! find_height - property that points to routines for determining status window height
property find_height alias u_to
! draw_window - routines with instructions for drawing the status window
property draw_window alias e_to
!\ bottom_justified - have this return true for status windows where regular
status information shares a window with other information and you want the
regular status information to be printed at the bottom of the window \!
property bottom_justified alias d_to
! terp_type - this property gets set to the current interpreter type automatically
property terp_type alias w_to
!\ status_override - normally, the status window object with the highest
find_height number gets drawn (so have those properties return 0 when not in
use), but sometimes more than one extension *could* be drawn, so we use
status_override to have one override the other (have it return true to initiate
such override) \!
property status_override alias nw_to
!\ chosen_window - set to the window instructions object whose draw_window
property will be executed. authors can ignore this. \!
property chosen_window alias nw_to
! "terp_type" values 0, 2, 4
enumerate step * 2
{
NORMAL_TERP, GLK_TERP = 2, SIMPLE_TERP
}
!\ a PrintStatusLine object in which we will put our instruction objects in (by
inclusion of extensions and what not. PrintStatusLine will call this directly).
\!
object printstatuslib
{
find_height
{
local highest, i, a
for i in self
{
a = i.find_height
if i.status_override
{
self.chosen_window = i
highest = i.find_height
break
}
if higher(highest,a) = a
{
self.chosen_window = i
highest = i.find_height
}
}
return highest
}
draw_window
{
run (self.chosen_window).draw_window
}
chosen_window 0
terp_type NORMAL_TERP
bottom_justified 1
}
replace PrintStatusline
{
local newstatusheight
! set the "terp_type" value
if IsGlk
printstatuslib.terp_type = GLK_TERP
elseif system(61) ! minimal port
printstatuslib.terp_type = SIMPLE_TERP
else
printstatuslib.terp_type = NORMAL_TERP
#ifset CHEAP
if cheap and printstatuslib.terp_type ~= SIMPLE_TERP
{
#if defined GAME_TITLE
CenterTitle(GAME_TITLE,0,1)
#endif
#if undefined GAME_TITLE
CenterTitle(CheapTitle,0,1)
#endif
display.needs_repaint = false
return
}
elseif cheap
return
#endif ! CHEAP
! figure out the size our window will be
newstatusheight = printstatuslib.find_height
! clear/remove the window if the status window height has changed
if (newstatusheight < display.statusline_height) and not system(61)
{
window display.statusline_height
{cls} ! clear whatever's there
window 0
}
display.statusline_height = newstatusheight
Font(BOLD_OFF | ITALIC_OFF | UNDERLINE_OFF | PROP_OFF)
window display.statusline_height
{
if printstatuslib.terp_type ~= SIMPLE_TERP
{
color SL_TEXTCOLOR, SL_BGCOLOR
cls
locate 1,1
}
run printstatuslib.draw_window
}
color TEXTCOLOR, BGCOLOR, INPUTCOLOR
Font(DEFAULT_FONT)
}
! Here is an example status window instruction object. It and its routines
! draw a regular status window
object statuswindow
{
in printstatuslib
find_height
{
return (call &FindStatusHeight)
}
draw_window
{
return (call &WriteStatus)
}
}
!\ Note: These properties *could* just say "return FindStatus". I just used
the above syntax to give a clue as to how one would change a value "mid-game".
If you want statuswindow.find_height to point to *another* routine, you could
have a line like this:
statuswindow.find_height = call &FindNewStatusHeight \!
! routine for finding the height of the regular status info
routine FindStatusHeight
{
local a, b
text to _temp_string
! can't start off a string with a space, it seems
!if not location ! so we'll save this space-writing code for the
! print "\_"; ! "status-writing" routine
!else
if not light_source
print "In the dark";
else
{
print capital location.name;
if FORMAT & DESCFORM_F
print "\_";
}
text to 0
a = StringLength(_temp_string)
text to _temp_string
select STATUSTYPE
case 1 : print number score; " / "; number counter;
case 2 : print HoursMinutes(counter);
case 3 : print "Score: "; number score; "\_ "; "Moves: "; number counter;
! STATUSTYPE case 3 is the "Infocom"-style status
case 4 : StatusType4 ! routine for configurable statusline
if (FORMAT & DESCFORM_F) and (printstatuslib.terp_type ~= GLK_TERP)
print "\_";
text to 0
if STATUSTYPE
b = StringLength(_temp_string)
if (b + a + 4)<display.screenwidth ! let's force a 4 character gap between
{ ! the two fields
return 1
}
elseif (b + a - 4 ) < display.screenwidth and STATUSTYPE = 3
{
text to _temp_string
print "S: "; number score; "\_ "; "M: "; number counter;
if (FORMAT & DESCFORM_F) and (printstatuslib.terp_type ~= GLK_TERP)
print "\_";
text to 0
return 1
}
else
return 2
}
! Roody's note: Replace this if you want to use the top right area
! for something else ("HUNGRY", "TIRED", or whatever)
routine STATUSTYPE4
{}
! routine for drawing the regular status
routine WriteStatus
{
if printstatuslib.bottom_justified and
printstatuslib.terp_type ~= SIMPLE_TERP
{
if statuswindow.find_height = 2
{
locate 1, (display.windowlines - 1)
}
else
locate 1, display.windowlines
}
if not location
print "\_";
elseif not light_source
print "In the dark";
else
{
if FORMAT & DESCFORM_F or (printstatuslib.terp_type = GLK_TERP)
print "\_";
print capital location.name;
}
if statuswindow.find_height = 1 and STATUSTYPE
{
print to (display.linelength - \
(StringLength(_temp_string) + \
((printstatuslib.terp_type = SIMPLE_TERP)*2) ));
StringPrint(_temp_string)
}
elseif STATUSTYPE and statuswindow.find_height = 2
{
if printstatuslib.terp_type ~= SIMPLE_TERP and
not printstatuslib.bottom_justified
locate 1, 2
else
""
if (FORMAT & DESCFORM_F) or (printstatuslib.terp_type = GLK_TERP)
print "\_";
StringPrint(_temp_string)
}
}
Showing posts with label PrintStatusLine. Show all posts
Showing posts with label PrintStatusLine. Show all posts
Wednesday, September 26, 2012
More PrintStatusLine stuff
Today, I got around to applying my new PrintStatusLine design to status-line-changing library extensions like newautomap.h and newconverse.h. It took some tweaking, but now with the current version of RoodyLib, it involves no finessing at all to have glk-enabled automap working in a game along with top-screen conversation menus. Just because I feel like my PrintStatusLine core code has changed just enough, I'm going to share the entire thing again. It might make some more sense if you are looking at a status-window-drawing extension along with this (like newautomap.h in http://roody.gerynarsabode.org/hbe/newautomap.zip), but maybe you can get some idea through the comments below:
Monday, September 17, 2012
What's my PrintStatusLine?
I've been pretty good lately about coming up with roodylib improvements instead of working on my game. The last idea I had was especially challenging, and I have to admit, it took a couple days of pecking to truly get up the steam to finish it.
Now, I probably could have done it all with multiple routines and maybe some global variables and arrays, but like my other "settings" in roodylib, I decided to make it object based. Extension PrintStatusLine objects have properties that point to routines that determine the height they need and instructions for drawing their portion of the status window.
Even when you aren't using fancy library extension stuff, you could make the status window bigger than usual and draw whatever you want in the extra space (and you can make the normal status part of the window "bottom justified" if you want).
I just finished this thing so hopefully I'll be able to pretty it up a little so it's easier to understand at some point, but it's uncertain that this version of PrintStatusLine will be readable to anyone but me, unfortunately. Let's take a look at what I wrote:
So, say, if you're using newconverse's version of this (after I update it, that is), it'd point printstatuslib find_height and draw_window properties to either these routines or newconverse's status window instructions, depending on whether the player is in a conversation.
Hopefully, this isn't too complicated.
In a nutshell..
I've been annoyed that some of my library extensions have to replace PrintStatusLine. What do you do when two of your extensions replace PrintStatusLine? Of course, you hack together a version that works this way in some circumstances and that way in others. Like other parts of roodylib, I wondered if there wasn't a way to make it more modular.Now, I probably could have done it all with multiple routines and maybe some global variables and arrays, but like my other "settings" in roodylib, I decided to make it object based. Extension PrintStatusLine objects have properties that point to routines that determine the height they need and instructions for drawing their portion of the status window.
Even when you aren't using fancy library extension stuff, you could make the status window bigger than usual and draw whatever you want in the extra space (and you can make the normal status part of the window "bottom justified" if you want).
I just finished this thing so hopefully I'll be able to pretty it up a little so it's easier to understand at some point, but it's uncertain that this version of PrintStatusLine will be readable to anyone but me, unfortunately. Let's take a look at what I wrote:
property find_height alias u_to
property draw_window alias e_to
property bottom_justified alias d_to
property terp_type alias w_to
! "terp_type" values 0, 2, 4
enumerate step * 2
{
NORMAL_TERP, GLK_TERP = 2, SIMPLE_TERP
}
object printstatuslib
{
find_height
{
local sum, i
for i in self
{
sum += i.find_height
}
return sum
}
draw_window
{
local i
for i in self
{
run i.draw_window
}
}
terp_type NORMAL_TERP
bottom_justified 0
}
! this object and its properties draw the status window as we normally know it
object statuswindow
{
in printstatuslib
find_height
{
return (call &FindStatusHeight)
}
draw_window
{
return (call &WriteStatus)
}
}
replace PrintStatusline
{
local newstatusheight
#ifset CHEAP
if cheap
return
#endif
! set the "terp_type" value
if IsGlk
printstatuslib.terp_type = GLK_TERP
elseif system(61) ! minimal port
printstatuslib.terp_type = SIMPLE_TERP
else
printstatuslib.terp_type = NORMAL_TERP
! figure out the size our window will be
newstatusheight = printstatuslib.find_height
! remove the windows if the status window height has changed
if (newstatusheight < display.statusline_height) and not system(61)
window 0
display.statusline_height = newstatusheight
Font(BOLD_OFF | ITALIC_OFF | UNDERLINE_OFF | PROP_OFF)
window display.statusline_height
{
color SL_TEXTCOLOR, SL_BGCOLOR
if printstatuslib.terp_type ~= SIMPLE_TERP
{
cls
locate 1,1
}
run printstatuslib.draw_window
}
color TEXTCOLOR, BGCOLOR, INPUTCOLOR
Font(DEFAULT_FONT)
}
! routine for finding the height of the regular status info
replace FindStatusHeight
{
local a, b
text to _temp_string
if not location
print "\_";
elseif not light_source
print "In the dark";
else
{
if FORMAT & DESCFORM_F: print "\_";
print capital location.name;
print "\_";
}
text to 0
a = StringLength(_temp_string)
if STATUSTYPE = 1
{
text to _temp_string
if (FORMAT & DESCFORM_F)
print "\_";
print number score; " / "; number counter;
if (FORMAT & DESCFORM_F)
print "\_";
text to 0
}
elseif STATUSTYPE = 3
{
text to _temp_string
if (FORMAT & DESCFORM_F) : print "\_";
print "Score: "; number score; "\_ "; "Moves: "; number counter;
if (FORMAT & DESCFORM_F) : print "\_";
text to 0
b = StringLength(_temp_string)
}
elseif STATUSTYPE = 2
{
text to _temp_string
if (FORMAT & DESCFORM_F) : print "\_";
print HoursMinutes(counter);
if (FORMAT & DESCFORM_F) : print "\_";
text to 0
}
elseif STATUSTYPE = 4
{
text to _temp_string
if (FORMAT & DESCFORM_F) : print "\_";
STATUSTYPE4 ! routine for configurable statusline
if (FORMAT & DESCFORM_F) : print "\_";
text to 0
}
if (b + a + 4)<display.screenwidth ! let's force a 4 character gap between
{ ! the two fields
return 1
}
elseif (b + a - 4 ) < display.screenwidth and STATUSTYPE = 3
{
text to _temp_string
if (FORMAT & DESCFORM_F) : print "\_";
print "S: "; number score; "\_ "; "M: "; number counter;
if (FORMAT & DESCFORM_F) : print "\_";
text to 0
return 1
}
else
return 2
}
! Roody's note: Replace this if you want to use the top right area
! for something else ("HUNGRY", "TIRED", or whatever)
replace STATUSTYPE4
{}
! routine for drawing the regular status
routine WriteStatus
{
if printstatuslib.bottom_justified and
printstatuslib.terp_type ~= SIMPLE_TERP
{
if statuswindow.find_height = 2
{
locate 1, (display.windowlines - 1)
}
else
locate 1, display.windowlines
}
if not location
print "\_";
elseif not light_source
print "In the dark";
else
{
if FORMAT & DESCFORM_F: print "\_";
print capital location.name;
}
if statuswindow.find_height = 1 and STATUSTYPE
{
print to (display.screenwidth - \
(StringLength(_temp_string) + \
((printstatuslib.terp_type = SIMPLE_TERP)*2)));
StringPrint(_temp_string)
}
elseif STATUSTYPE and statuswindow.find_height = 2
{
if printstatuslib.terp_type ~= SIMPLE_TERP and
not printstatuslib.bottom_justified
locate 1, 2
else
""
StringPrint(_temp_string)
}
}
So, say, if you're using newconverse's version of this (after I update it, that is), it'd point printstatuslib find_height and draw_window properties to either these routines or newconverse's status window instructions, depending on whether the player is in a conversation.
Hopefully, this isn't too complicated.
One other thing...
I also threw together an extension for score notification, as I never have these things around when I need them. I'll add proper documentation and upload it somewhere at some point:!::Actually, I haven't even compiled that code yet, so there might be a typo or other kind of bug in there.
! Hugo Score Notification extension
!::
!\
Provides text like "You score has gone up by [x] points!"
\!
#ifclear _SCORENOTIFY_H
#SET _SCORENOTIFY_H
#ifset VERSIONS
#message "ScoreNotify.h Version 0.5"
#endif
property score_notify alias d_to
property points alias e_to
object scorenotifylib "scorenotify"
{
score_notify true
points 0
#ifset _ROODYLIB_H
save_info
{
select self.score_notify
case 0 : SaveWordSetting("score_off")
case 1 : SaveWordSetting("score_on")
return true
}
type settings
in init_instructions
execute
{
local a
a = CheckWordSetting("scorenotify")
if a
{
select word[(a-1)]
case "score_off": self.score_notify = 0
case "score_on": self.score_notify = 1
}
}
#endif
#ifset _NEWMENU_H
usage_desc
{
"\BSCORE NOTIFICATION ON\b- Be notified when you score points."
Indent
"\BSCORE NOTIFICATION OFF\b- Play without score notifications."
}
#endif ! NEWMENU
}
#ifset _ROODYLIB_H
object scorenotifymain
{
type settings
in main_instructions
execute
{
ScoreNotify
}
}
#endif ! _ROODYLIB_H
routine ScoreNotify
{
if scorenotifylib.points and scorenotifylib.score_notify
{
""
Font(BOLD_ON)
ScoreNotificationMessage(&ScoreNotify, 1, scorenotifylib.points ) ! "[Your score has gone up.]"
Font(BOLD_OFF
}
score += scorenotifylib.points ! add the points to the score
scorenotifylib.points = 0 ! reset the point counter
}
! routine to call for the last score of a game (after the winning move), as
! main is not called again
routine LastScore(a)
{
score += a
}
! otherwise, call this routine to add to the game score
routine AddScore(a)
{
scorenotifylib.points += a
}
routine DoNotifyOnOff
{
if scorenotifylib.score_notify
Perform(&DoNotifyOff)
else
Perform(&DoNotifyOn)
}
routine DoNotifyOn
{
if scorenotifylib.score_notify
ScoreNotificationMessage(&DoNotifyOn, 1 ) ! "[Score notification already on.]"
else
{
ScoreNotificationMessage(&DoNotifyOn, 2 ) ! "[Score notification on.]"
scorenotifylib.score_notify = 1
}
}
routine DoNotifyOff
{
if not scorenotifylib.score_notify
ScoreNotificationMessage(&DoNotifyOff, 1 ) ! "[Score notification already off.]"
else
{
ScoreNotificationMessage(&DoNotifyOff, 2 ) ! "[Score notification off.]"
scorenotifylib.score_notify = 0
}
}
routine ScoreNotificationMessage(r, num, a, b)
{
if NewScoreNotificationMessages(r, num, a, b): return
select r
case &DoNotifyOn
{
select num
case 1: "[Score notification already on.]"
case 2: "[Score notification on.]"
}
case &DoNotifyOff
{
select num
case 1: "[Score notification already off.]"
case 2: "[Score notification off.]"
}
case &ScoreNotify
{
select num
case 1 : "[Your score has gone up.]"
}
}
!\ The NewScoreNotificationMessages routine may be REPLACED and should return
true if a replacement message exists for routine <r> \!
routine NewScoreNotificationMessages(r, num, a, b)
{
select r
! case &ScoreNotify
! {
! select num
! case 1
! {
! print "[Your score has gone up by "; number a; " points.]"
! }
! }
case else : return false
return true ! this line is only reached if we replaced something
}
#endif _SCORENOTIFY_H
Subscribe to:
Comments (Atom)