1998-12-02 20:58:08 +01:00
|
|
|
/* generic.c */
|
1999-02-14 12:15:47 +01:00
|
|
|
/* Copyright 1999 - Joseph Pranevich */
|
1998-12-02 20:58:08 +01:00
|
|
|
|
|
|
|
/* This is a driver to implement, when possible, "high-level"
|
|
|
|
routines using only low level calls. This is to make it possible
|
|
|
|
to have accelerated functions for the individual drivers...
|
|
|
|
or to simply not bother with them. */
|
|
|
|
|
|
|
|
/* When creating new drivers, you need to assign all the functions that
|
|
|
|
that driver supports into the driver struct. If it is a supplementary
|
2000-12-27 05:02:46 +01:00
|
|
|
driver, it should make sure to preserve the old values. */
|
1998-12-02 20:58:08 +01:00
|
|
|
|
|
|
|
#include <stdio.h>
|
1998-12-09 16:43:03 +01:00
|
|
|
|
1998-12-02 20:58:08 +01:00
|
|
|
#include "console.h"
|
|
|
|
#include "config.h"
|
1999-05-14 10:17:14 +02:00
|
|
|
#include "debugtools.h"
|
1998-12-02 20:58:08 +01:00
|
|
|
|
2000-11-28 00:54:25 +01:00
|
|
|
DEFAULT_DEBUG_CHANNEL(console);
|
1999-04-19 16:56:29 +02:00
|
|
|
|
1999-01-03 17:15:12 +01:00
|
|
|
static void GENERIC_MoveLine(char row1, char row2, char col1, char col2);
|
|
|
|
static void GENERIC_ClearLine(char row, char col1, char col2, int bgcolor,
|
|
|
|
int attribute);
|
2000-04-09 20:40:32 +02:00
|
|
|
void GENERIC_Start(void)
|
1998-12-02 20:58:08 +01:00
|
|
|
{
|
|
|
|
/* Here, we only want to add a driver if there is not one already
|
|
|
|
defined. */
|
|
|
|
|
1999-05-14 10:17:14 +02:00
|
|
|
TRACE("GENERIC_Start\n");
|
1998-12-02 20:58:08 +01:00
|
|
|
|
|
|
|
if (!driver.clearWindow)
|
|
|
|
driver.clearWindow = GENERIC_ClearWindow;
|
|
|
|
|
|
|
|
if (!driver.scrollUpWindow)
|
|
|
|
driver.scrollUpWindow = GENERIC_ScrollUpWindow;
|
|
|
|
|
|
|
|
if (!driver.scrollDownWindow)
|
|
|
|
driver.scrollDownWindow = GENERIC_ScrollDownWindow;
|
|
|
|
|
|
|
|
if (!driver.getCharacter)
|
|
|
|
driver.getCharacter = GENERIC_GetCharacter;
|
|
|
|
}
|
|
|
|
|
|
|
|
void GENERIC_ClearWindow(char row1, char col1, char row2, char col2,
|
|
|
|
int bg_color, int attribute)
|
|
|
|
{
|
1999-01-03 17:15:12 +01:00
|
|
|
char trow, tcol, x;
|
1998-12-02 20:58:08 +01:00
|
|
|
int old_refresh;
|
|
|
|
|
|
|
|
/* Abort if we have only partial functionality */
|
|
|
|
if (!(driver.getCursorPosition && driver.moveCursor && driver.write))
|
|
|
|
return;
|
|
|
|
|
|
|
|
old_refresh = CONSOLE_GetRefresh();
|
|
|
|
CONSOLE_SetRefresh(FALSE);
|
|
|
|
|
|
|
|
CONSOLE_GetCursorPosition(&trow, &tcol);
|
|
|
|
|
1999-01-03 17:15:12 +01:00
|
|
|
for (x = row1; x <= row2; x++)
|
|
|
|
GENERIC_ClearLine(x, col1, col2, bg_color, attribute);
|
1998-12-02 20:58:08 +01:00
|
|
|
|
|
|
|
CONSOLE_MoveCursor(trow, tcol);
|
|
|
|
|
|
|
|
CONSOLE_SetRefresh(old_refresh);
|
|
|
|
}
|
|
|
|
|
|
|
|
void GENERIC_ScrollUpWindow(char row1, char col1, char row2, char col2,
|
|
|
|
char lines, int bg_color, int attribute)
|
|
|
|
{
|
1999-01-03 17:15:12 +01:00
|
|
|
/* Scroll Up Window: Characters go down */
|
|
|
|
|
1999-07-31 19:36:48 +02:00
|
|
|
char trow, tcol, x;
|
|
|
|
int old_refresh;
|
1998-12-02 20:58:08 +01:00
|
|
|
|
1999-05-14 10:17:14 +02:00
|
|
|
TRACE("Scroll Up %d lines from %d to %d.\n", lines, row1,
|
1999-01-03 17:15:12 +01:00
|
|
|
row2);
|
1998-12-02 20:58:08 +01:00
|
|
|
|
|
|
|
/* Abort if we have only partial functionality */
|
|
|
|
if (!(driver.getCursorPosition && driver.moveCursor && driver.write
|
|
|
|
&& driver.getCharacterAtCursor && driver.clearWindow))
|
|
|
|
return;
|
1999-01-03 17:15:12 +01:00
|
|
|
|
|
|
|
/* Save initial state... */
|
1998-12-02 20:58:08 +01:00
|
|
|
old_refresh = CONSOLE_GetRefresh();
|
|
|
|
CONSOLE_SetRefresh(FALSE);
|
|
|
|
CONSOLE_GetCursorPosition(&trow, &tcol);
|
1999-01-03 17:15:12 +01:00
|
|
|
|
|
|
|
for (x = row1 + lines; x <= row2; x++)
|
1998-12-02 20:58:08 +01:00
|
|
|
{
|
1999-01-03 17:15:12 +01:00
|
|
|
GENERIC_MoveLine(x, x - lines, col1, col2);
|
|
|
|
GENERIC_ClearLine(x, col1, col2, bg_color, attribute);
|
|
|
|
}
|
1998-12-02 20:58:08 +01:00
|
|
|
|
1999-01-03 17:15:12 +01:00
|
|
|
/* Restore State */
|
|
|
|
CONSOLE_MoveCursor(trow, tcol);
|
1998-12-02 20:58:08 +01:00
|
|
|
CONSOLE_SetRefresh(old_refresh);
|
|
|
|
}
|
|
|
|
|
|
|
|
void GENERIC_ScrollDownWindow(char row1, char col1, char row2, char col2,
|
|
|
|
char lines, int bg_color, int attribute)
|
|
|
|
{
|
1999-01-03 17:15:12 +01:00
|
|
|
/* Scroll Down Window: Characters go up */
|
1998-12-02 20:58:08 +01:00
|
|
|
|
1999-07-31 19:36:48 +02:00
|
|
|
char trow, tcol, x;
|
|
|
|
int old_refresh;
|
1998-12-02 20:58:08 +01:00
|
|
|
|
|
|
|
/* Abort if we have only partial functionality */
|
|
|
|
if (!(driver.getCursorPosition && driver.moveCursor && driver.write
|
|
|
|
&& driver.getCharacterAtCursor && driver.clearWindow))
|
|
|
|
return;
|
1999-01-03 17:15:12 +01:00
|
|
|
|
|
|
|
/* Save initial state... */
|
1998-12-02 20:58:08 +01:00
|
|
|
old_refresh = CONSOLE_GetRefresh();
|
|
|
|
CONSOLE_SetRefresh(FALSE);
|
|
|
|
CONSOLE_GetCursorPosition(&trow, &tcol);
|
1999-01-03 17:15:12 +01:00
|
|
|
|
|
|
|
for (x = row2; x >= row1 + lines; x--)
|
1998-12-02 20:58:08 +01:00
|
|
|
{
|
1999-01-03 17:15:12 +01:00
|
|
|
GENERIC_MoveLine(x, x + lines, col1, col2);
|
|
|
|
GENERIC_ClearLine(x, col1, col1, bg_color, attribute);
|
|
|
|
}
|
1998-12-02 20:58:08 +01:00
|
|
|
|
1999-01-03 17:15:12 +01:00
|
|
|
/* Restore State */
|
|
|
|
CONSOLE_MoveCursor(trow, tcol);
|
1998-12-02 20:58:08 +01:00
|
|
|
CONSOLE_SetRefresh(old_refresh);
|
|
|
|
}
|
|
|
|
|
|
|
|
char GENERIC_GetCharacter()
|
|
|
|
{
|
|
|
|
/* Keep getting keys until we get one with a char value */
|
|
|
|
char ch = (char) 0, scan;
|
|
|
|
|
|
|
|
while (!ch)
|
|
|
|
{
|
1999-01-17 17:55:11 +01:00
|
|
|
CONSOLE_GetKeystroke(&scan, &ch);
|
1998-12-02 20:58:08 +01:00
|
|
|
}
|
|
|
|
return ch;
|
|
|
|
}
|
|
|
|
|
1999-01-03 17:15:12 +01:00
|
|
|
static void GENERIC_ClearLine(char row, char col1, char col2, int bgcolor,
|
|
|
|
int attribute)
|
|
|
|
{
|
|
|
|
/* This function is here to simplify the logic of the scroll and clear
|
|
|
|
functions but may be useful elsewhere. If it can be used from
|
|
|
|
outside here, it should be made non-static */
|
|
|
|
|
1999-07-31 19:36:48 +02:00
|
|
|
char x;
|
1999-01-03 17:15:12 +01:00
|
|
|
|
2000-11-11 01:38:37 +01:00
|
|
|
TRACE("Clear Line: %d from %d to %d (unused: bgcolor %d, attrib %d).\n", row, col1, col2, bgcolor, attribute);
|
1999-01-03 17:15:12 +01:00
|
|
|
|
|
|
|
for (x = col1; x <= col2; x++)
|
|
|
|
{
|
|
|
|
CONSOLE_MoveCursor(row, x);
|
|
|
|
CONSOLE_Write(' ', 0, 0, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Assume that the calling function will make sure that the cursor is
|
|
|
|
repositioned properly. If this becomes non-static, that will need to be
|
|
|
|
changed. */
|
|
|
|
}
|
|
|
|
|
|
|
|
static void GENERIC_MoveLine(char row1, char row2, char col1, char col2)
|
|
|
|
{
|
|
|
|
/* This function is here to simplify the logic of the scroll and clear
|
|
|
|
functions but may be useful elsewhere. If it can be used from
|
|
|
|
outside here, it should be made non-static */
|
|
|
|
|
1999-07-31 19:36:48 +02:00
|
|
|
char x;
|
1999-01-03 17:15:12 +01:00
|
|
|
int bg_color, fg_color, attribute;
|
|
|
|
char ch;
|
|
|
|
|
1999-05-14 10:17:14 +02:00
|
|
|
TRACE("Move Line: Move %d to %d.\n", row1, row2);
|
1999-01-03 17:15:12 +01:00
|
|
|
|
|
|
|
for (x = col1; x <= col2; x++)
|
|
|
|
{
|
|
|
|
CONSOLE_MoveCursor(row1, x);
|
|
|
|
CONSOLE_GetCharacterAtCursor(&ch, &fg_color, &bg_color, &attribute);
|
|
|
|
CONSOLE_MoveCursor(row2, x);
|
|
|
|
CONSOLE_Write(ch, fg_color, bg_color, attribute);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Assume that the calling function will make sure that the cursor is
|
|
|
|
repositioned properly. If this becomes non-static, that will need to be
|
|
|
|
changed. */
|
|
|
|
}
|