ЭЛЕКТРОННАЯ БИБЛИОТЕКА КОАПП
Сборники Художественной, Технической, Справочной, Английской, Нормативной, Исторической, и др. литературы.



 

Часть 5

                                  ГЛАВА 4                     -- 1 --
                                  -------

                                  ГРАФИКА
     -----------------------------------------------------------------

          В этой  главе  приводится  базовый  набор  функций  графики,
     которые позволяют рисовать (отображать на экране)  точки,  линии,
     прямоугольники,  окружности,  используя  возможности  графических
     адаптеров CGA или EGA.  Эти программы используются как основа, на
     которой строятся функции графики более высокого уровня. Для более
     детального изучения этих базовых графических функций предлагается
     книга   "С:   The  Complete  Reference"  Herb  Schild  (Osborn  /
     McGrow-Hall, 1987).

          Помимо краткого представления  базовых  функций  графики,  в
     этой главе приводятся следующие программы:

          - сохранение графических изображений в файле на диске;
          - загрузка графических изображений из файла;
          - вращение объектов в двумерном пространстве;
          - копирование или пересылка графических изображений.

          В   конце   главы   привевен   текст  "программы-художника",
     позволяющей рисовать на экране терминала, с использованием клавиш
     перемещения курсора.

          По мере возрастания вашего опыта  по  использованию  функций
     графики,   вы   сможете   самостоятельно   разрабатывать  хорошие
     программы.  Например,  используя функции  сохранения  и  загрузки
     графических   изображений,   вы  сможете  создавать  графики  или
     диаграммы и  успешно  использовать  их  в  случае  необходимости.
     Используя  функции  вращения  изображений  объектов,  вы  сможете
     разрабатывать  программы  "звездных  войн"  -   образец   "живой"
     графики,  которые будут представлять большой интерес для вас.  Вы
     сможете  также,  использовать  эти   функции   как   основу   для
     использования  систем автоматизированного проектирования (CAD/CAM
     system).

          Для  корректного  использования  программ,  описаных  в этой
     главе,  вам необходимы компьютеры типа IBM PC  XT,  IBM PC AT или
     другие    совместимые  с  ними  модели,  снабженные  графическими
     адаптерами CGA  или  EGA.  Все  программы,  приведенные  в данной
     главе,  кроме программ изображения точки, аппаратно-независимы, и
     вы можете с минимальными усилиями  сделать их  рабочими на других
     типах графических машин.













                                 "C" для профессиональных программистов
Глава IV                                                       -- 2 --


                      ВИДЕОРЕЖИМЫ  И  ЦВЕТОВАЯ ПАЛИТРА
     -----------------------------------------------------------------

          Перед использованием  каких-либо  функций  графики компьютер
     должен  быть   переведен   в   соответствующий   видеорежим.  Для
     компьютеров  типа  IBM PC  это означает,  что должны быть выбраны
     подходящие режим и палитра.

          В  таблице 4-1  приведены различные  видеорежимы,  в которых
     могут работать  компьютеры  IBM PC.  Для  функций,  приведенных в
     этой главе,  требуется 4 видеорежим, предполагающий использование
     цветного графического дисплея  с размерностью экрана 320  на 200.
     Хотя адаптер EGA поддерживает и режимы  с расширенной разрешающей
     способностью дисплея, 4 видеорежим выбран в качестве базового для
     разработки и использования функций графики в связи с тем,  что он
     поддерживается как адаптером EGA, так и CGA. Использование особых
     режимов EGA требует только изменения функций записи точки (смотри
     книгу  по  использованию графики в  EGA  "Advance  Grafics  in C"
     Nelson   Jobson   Osborn/McGrow-Hall,   1987).   Вам   необходимо
     запомнить,  что во всех кодах верхний левый угол имеет координаты
     0,0.

          BIOS-прерывание 16,  функция  0,  устанавливает видеорежим и
     используется в функции mode(),  текст которой приведен  ниже.  Но
     прежде  чем  вы  ознакомитесь  с ней,  предлагаем вам внимательно
     изучить   все   видеорежимы,   поддерживаемые    соответствующими
     адаптерами. Для этого обратитесь к таблице 4-1.


     Таблица 4-1 Режимы терминала для машин IBM PC

     -----------------------------------------------------------------
      Режим             Тип            Размерность  Адаптер
                                         экрана
     -----------------------------------------------------------------

        0    алфавитно-цифровой, ч/б      40х25    CGA, EGA
        1    алфавитно-цифровой, 16 цв.   40х25    CGA, EGA
        2    алфавитно-цифровой, ч/б      80х25    CGA, EGA
        3    алфавитно-цифровой, 16 цв.   80х25    CGA, EGA
        4    графический, 4 цвета       320х200    CGA, EGA
        5    графический, 4 серых тона  320х200    CGA, EGA
        6    графический, ч/б           640х200    CGA, EGA
        7    алфавитно-цифровой, ч/б      80х25   монохромный
        8    графический, 16 цветов     160х200     PCjr
        9    графический, 16 цветов     320х200     PCjr
       10    графический, 4 цвета-PCjr  640х200    PCjr, EGA
                         16 цветов-EGA
       13    графический, 16 цветов     320х200      EGA
       14    графический, 16 цветов     640х200      EGA
       15    графический, 4 цвета       640х350      EGA
     -----------------------------------------------------------------

          А теперь приведем текст функции mode().


                                 "C" для профессиональных программистов
Глава IV                                                       -- 3 --



          /* Установка видеорежима */
          void mode(mode_code)
          int mode_code;
               {
               union REGS r;
               r.h.al = mode_code;
               r.h.ah = 0;
               int86(0x10,&r,&r);
               }


          В 4-ом режиме доступны  две палитры (набора  цветов). Каждая
     палитра   определяет  четыре   цвета,   отображаемые   на  экране
     терминала.  В  IBM PC  палитра  0  определяет  желтый,  зеленый и
     красный   цвета,   палитра  1   определяет   белый,  ярко-красный
     (пурпурный)  и голубой цвета. Для каждой палитры четвертым цветом
     является цвет  фона,  который обычно черный.  BIOS-прерывание 16,
     функция 11, устанавливает палитру.

          Функция  pallet(),  приведенная  ниже,  устанавливает  номер
     палитры, который задается в качестве значения ее аргумента:

          /* Установка палитры */
          void palette(pnum)
          int pnum;
               {
               union REGS r;

               r.h.bh=1;    /* код 4 графического режима */
               r.h.bl=pnum; /* номер палитры */
               r.h.ah=11;   /* устанавливается для вызова палитры */
               int86(0x10,&r,&r);
               }






















                                 "C" для профессиональных программистов
Глава IV                                                       -- 4 --


                           ЗАПИСЬ  ТОЧКИ  РАСТРА
     -----------------------------------------------------------------

          Одной из основных программ графики является программа записи
     точки растра - наименьшей адресуемой точки на экране  дисплея.  В
     данном  случае  термин  "точка  растра"  будет использоваться для
     описания наименьшей  адресуемой  точки  в  отдельных  графических
     режимах.  Так  как  функция  записи  точки  растра используется в
     других программах более высокого уровня,  ее эффективность  очень
     важна   для  быстродействия  программ  верхних  уровней,  которые
     непосредственно  реализуют  функции  графики.   На   IBM   PC   и
     совместимых  с  ними  компьютерах  существуют  два способа вывода
     информации  с  использованием  точек   растра.   Первый   способ,
     использующий прерывания ROM-BIOS, является наиболее простым, но и
     наименее быстродействующим (слишком медленным для  наших  целей).
     Вторым  и  более  быстродействующими способом является размещение
     информации непосредственно в видеопамяти  дисплея  (ROM).  Именно
     этот метод и рассматривается ниже.






































                                 "C" для профессиональных программистов
Глава IV                                                       -- 5 --


     Работа адаптеров CGA/EGA в графическеом режиме
     ----------------------------------------------

          Адаптер CGA  всегда  располагается  в  видеопамяти по адресу
     8000000h.  Адаптер EGA имеет  аналогичное  расположение  для  тех
     режимов,   которые   совместимы  с  режимами  CGA  (более  полную
     информацию о аппаратных средствах  поддержки  графики  вы  можете
     получить   в   руководстве   "IBM   Technical  Reference").  В  4
     графическом режиме каждый байт содержит информацию о цвете  для 4
     точек  растра (для каждой точки растра по 2 бита). Следовательно,
     для работы с  экраном  размерностью  320  на  200  требуется  16К
     памяти.  Так  как  два  бита  могут  содержать только 4 различных
     значения, в 4 видеорежиме поддерживаются только 4 цвета. Значение
     каждого  двухбитового  блока  определяет  цвет  в  соответствии с
     таблицей, приведенной ниже:

     ---------------------------------------------------
      Значение   Цвет в палитре 0     Цвет в палитре 1
     ---------------------------------------------------
                      фон                   фон
         1          желтый                голубой
         2          красный               пурпурный
         3          зеленый               булый
     ---------------------------------------------------

          Особенность адаптера CGA заключается в том, что четные точки
     растра будут располагаться по адресу B8000000h,  а нечетные -  на
     2000h  (8152  -  в десятичном виде) байтов выше,  т.е.  по адресу
     B8002000h.  Следовательно,  каждая строка точек растра требует 80
     байтов  (40 для четных точек растра,  40 - для нечетных).  Внутри
     каждого  байта  точки  располагаются  слева  направо,   как   они
     появляются на экране терминала.  Это означает, что точка растра с
     номером 0 занимает 6 и 7 биты,  в то время,  как точка  растра  с
     номером 3 - 0 и 1 биты.

          Так как  каждый байт кодирует значение четырех точек растра,
     вы должны сохранять значение трех точек при  изменении  одной  из
     них.  Лучший способ сделать это - создание битовой маски со всеми
     битами,  установленными в 1, кроме тех, которые будут изменяться.
     Значение битовой маски складывается по схеме "И" с действительным
     значением байта,  а затем  полученное  значение  складывается  по
     схеме   "ИЛИ"   с  новым  значением.  Однако  ситуация  несколько
     меняется, если вы хотите сложить по схеме "НЕ-ИЛИ" новое и старое
     значения.  В  этом  случае  вы  просто складываете по схеме "ИЛИ"
     старое значение байта с 0,  а затем складываете по схеме "НЕ-ИЛИ"
     новое  двоичное представление цвета и получаете результат.  Адрес
     требуемого байта определяется путем умножения значения координаты
     X на 40,  а затем добавляется значение координаты Y,  деленное на
     4.  Для того,чтобы определить, находится ли точка растра в четном
     или  нечетном  блоке  памяти,  используется  остаток  от  деления
     значения координаты Х на 2.  Если остаток равен 0,  блок является
     четным  (используется  первый  блок),  в  противном случае - блок
     нечетный (используется второй блок).  Требуемые биты внутри байта
     вычисляются   путем  выполнения  деления  по  модулю  4.  Остаток


                                 "C" для профессиональных программистов
Глава IV                                                       -- 6 --


     определяет номер двухбитового пакета, который содержит информацию
     о  требуемых  точках  растра.  Для установки байта режима цвета и
     битовой маски  используются  операторы  побитового  сдвига.  Хотя
     манипулирование  битами  в функции mempoint() несколько запутано,
     вы,  однако,  без труда разберетесь в ней, если тщательно изучите
     что именно и как она делает.


     /* Запись точки в CGA/EGA */
          void mempoint(x,y,color_code)
          int x,y,color_code;
            {
            union mask {
               char c[2];
               int i;
            } bit_mask;
            int i,index,bit_position;
            unsigned char t;
            char xor; /* "НЕ-ИЛИ" цвета в случае его изменения */
            char far *ptr=(char far *) 0xB8000000; /* точка в памяти
                                                                CGA */
            bit_mask.i=0xFF3F; /* 11111111 00111111 в двоичном коде */
            if (x<0 || x>199 || y<0 || y>319) return;
            xor=color_code & 128; /* проверка, устанавливался ли
                                     режим "НЕ-ИЛИ" */
            color_code=color_code & 127; /* маска старших битов */

            /*  установка битовой маски и битов режима цвета
                в правую позицию */

             bit_position=y%4; /* вычисление нужной позиции
                                  в байте */
             color_code<<=2*(3-bit_position); /* сдвиг кода цвета
                                        в нужную позицию */
             bit_mask.i>>=2*bit_position; /* сдвиг битовой маски в
                                          нужную позицию */

             /* определение требуемого байта в памяти терминала */
              index=x*40+(y%4);
              if (x%2) index+=8152; /* если нечетный, используется
                                       второй блок */
              /* запись цвета */
               if (!xor) {  /* режим изменения цвета */
                  t=*(ptr+index) & bit_mask.c[0];
                  *(ptr+index)=t|color_code;
               }
               else {
                  t=*(ptr+index) | (char)0;
                  *(ptr+index)=t & color_code;
               }
            }





                                 "C" для профессиональных программистов
Глава IV                                                       -- 7 --


          Заметим, что тип указателя видеопамяти объявлен как far; это
     необходимо, если  вы  транслируете  в  модели  маленькой  (small)
     памяти.  Вы так же должны заметить, что специальный маркер режима
     записи  XOR,  определенный в функциях ROM-BIOS,  используется и в
     функции mempoint().



















































                                 "C" для профессиональных программистов
Глава IV                                                       -- 8 --


                            ВЫЧЕРЧИВАНИЕ ЛИНИЙ
     -----------------------------------------------------------------

          Функции вычерчивания линий являются основными подпрограммами
     графики и используются для отображения  линий  в  заданном  цвете
     путем задания ее начальных и конечных координат.  В то время, как
     изображение вертикальных и горизонтальных линий  не  представляет
     особого  труда,  более трудной задачей является создание функций,
     которые рисуют линии  вдоль  диагоналей.  Например,  какие  точки
     составляют  линию,  вычерчиваемую  от точки с координатами 0,0 до
     точки с координатами 80,120?

          Один из подходов к  разработке  функций  вычерчивания  линий
     использует  отношение между смещением по координатам X и Y. Чтобы
     показать этот подход в действии,  проведем линию между точками  с
     координатами  0,0  и  5,10.  Смещение по X равно 5,  а по Y - 10.
     Отношение равно 1/2.  Оно  будет  использоватся  при  определении
     коэффициента зависимости,  по которому должны меняться координаты
     X и Y при изображении линий.  В данном случае это  означает,  что
     приращение  координаты  X  составляет половину величины изменения
     координаты Y.  Начинающий программист часто использует этот метод
     при  разработке  функций  вычерчивания  линий.  Хотя такой подход
     математически верен и прост для  понимания,  для  его  правильной
     работы  и  для того,  чтобы избежать серьезных ошибок округления,
     необходимо  использовать  математические  операции  с  числами  с
     плавающей  точкой.  Это означает,  что функции вычерчивания линий
     будут работать довольно медленно, если в систему не будет включен
     математический  сопроцессор  (например  -  Intel  8087).  По этой
     причине этот метод используется довольно редко.

          Наиболее общий    метод    изображения    линий     включает
     использование  алгоритма  Брезенхама.  Хотя  основой в нем служит
     также отношение между расстояниями по координатам X и Y, в данном
     случае  не  требуется  выполнять  деление  или вычисление чисел с
     плавающей  точкой.  Вместо  этого,  отношение  между   значениями
     координат  X  и  Y  представляется  косвенным образом через серии
     сложений  и  вычитаний.  Основной  идеей  алгоритма   Брезенхама,
     является   регистрация   средних   значений   погрешностей  между
     идеальным положением  каждой  точки  и  той  позицией  на  экране
     дисплея,  в  которой она действительно отображается.  Погрешность
     между идеальным и действительным положением точки возникает ввиду
     ограниченных  возможностей  технических  средств.  Фактически  не
     существует   дисплеев   с    бесконечно    большой    разрешающей
     способностью,  и,  следовательно, действительное положение каждой
     точки на линии требует наилучшей аппроксимации. В каждой итерации
     цикла  вычерчивания  линии вызываются две переменные xerr и yerr,
     которые  увеличиваются  в  зависимости   от   изменения   величин
     координат  X  и  Y  соответственно.  Когда  значение  погрешности
     достигает определенного значения,  оно  вновь  устанавливается  в
     исходное   положение,   а   соответствующий   счетчик   координат
     увеличивается.  Этот процесс продолжается до тех пор,  пока линия
     не будет полностью вычерчена.  Функция line(),  приведенная ниже,
     реализует этот метод.  Вы должны изучать ее до тех пор,  пока  не
     поймете механизма выполнения всех ее операций. Заметим, что в ней


                                 "C" для профессиональных программистов
Глава IV                                                       -- 9 --


     используется  функция   mempoint(),   разработанная   ранее   для
     отображения точки на экране терминала.


      /* Вычерчивание линии заданного цвета с использованием
         алгоритма Брезенхама */
       void line(startx,starty,endx,endy,color)
       int startx,starty,endx,endy,color;
       {
         register int t,distаnce;
         int xerr=0,yerr=0,delta_x,delta_y;
         int incx,incy;

       /* вычисление расстояния в обоих направлениях  */
         delta_x=endx-startx;
         delta_y=endy-starty;

       /* определение направления шага,
          шаг вычисляется либо по вертикальной, либо горизонтальной
          линии   */
          if (delta_x>0) incx=1;
          else  if (delta_x==0) incx=0;
          else  incx= -1;
          if (delta_y>0) incy=1;
          else  if (delta_y==0) incy=0;
          else  incy= -1;

        /* определение какое расстояние больше */
          delta_x=abs(delta_x);
          delta_y=abs(delta_y);
          if (delta_x>delta_y) distance=delta_x;
          else distance=delta_y;

        /* вычерчивание линии */
          for (t=0; t<=distance+1; t++) {
             mempoint(startx,starty,color);
             xerr+=delta_x;
             yerr+=delta_y;
             if (xerr>distance) {
                xerr-=distance;
                startx+=incx;
             }
             if (yerr>distance) {
                yerr-=distance;
                starty+=incy;
             }
          }
       }








                                 "C" для профессиональных программистов
Глава IV                                                       -- 10 --


                 ИЗОБРАЖЕНИЕ И ЗАКРАШИВАНИЕ ПРЯМОУГОЛЬНИКОВ
     -----------------------------------------------------------------

          Если у  вас есть функции вычерчивания линий,  то не составит
     особого  труда  создать  функции  вычерчивания   прямоугольников.
     Пример,  приведенный здесь, вычерчивает прямоугольники в заданном
     цвете путем задания координат двух противоположных углов.


         /*  Вычерчивание прямоугольника  */
        void box(startx,starty,endx,endy,color_code)
        int startx,starty,endx,endy,color_code;
        {
           line(startx,starty,endx,starty,color_code);
           line(startx,starty,startx,endy,color_code);
           line(startx,endy,endx,endy,color_code);
           line(endx,starty,endx,endy,color_code);
        }


          Для того,   чтобы   закрасить    прямоугольник,    требуется
     выполнить запись  в  каждую  точку  растра внутри прямоугольника.
     Программа    fill_box(),    приведенная     ниже,     закрашивает
     прямоугольник,  определенный  координатами  двух  противоположных
     углов,  заданным  цветом.  В  ней  используется  функция  line(),
     задающая цвет внутри прямоугольника.


        /* Закрашивание прямоугольника в заданный цвет */
        void fill_box(startx,starty,endx,endy,color_code)
        int startx,starty,endx,endy,color_code;
        {
           register int i,begin,end;

           begin=startxendx ? startx:endx;

           for (i=begin;i<=end;++i)
              line(i,starty,i,endy,color_code);
         }
















                                 "C" для профессиональных программистов
Глава IV                                                       -- 11 --


                          ВЫЧЕРЧИВАНИЕ ОКРУЖНОСТЕЙ
     -----------------------------------------------------------------

          Самым быстрым и  легким  способом  вычерчивания  окружностей
     является способ, основанный на использовании опять таки алгоритма
     Брезенхама,  похожего на одноименный алгоритм вычерчивания линий.
     Данный  метод  также  не  требует  вычислений  чисел  с плавающей
     точкой,  кроме  вычисления  коэффициента   сжатия,   поэтому   он
     обеспечивает  достаточное быстродействие.  По существу,  алгоритм
     основан на приращении координат X и  Y  на  величину  погрешности
     между  ними.  Значение  погрешности содержится в переменой delta.
     Функция  plot_circle()  выполняет  запись  точек  по  окружности.
     Переменная asp_ratio является глобальной,  т.к.  она используется
     как в функции  circle(),  так  и  в  функции  plot_circle().  Эта
     переменная   может   быть  полезна  с  точки  зрения  возможности
     установления ее  значения  вне  функции  circle()  и  дальнейшего
     использования  внутри  функции.  Путем  изменения  значения  этой
     переменной   вы  можете  рисовать  эллипсы.  Параметрами  функции
     circle() является  центр  окружности,  ее  радиус и цвет.  Тексты
     функций приведены ниже.


         double asp_ratio;

        /* Вычерчивание окружности с использованием алгоритма
           Брезенхама */
         void circle(x_center,y_center,radius,color_code)
         int x_center,y_center,radius,color_code;
         {
            register x,y,delta;
            asp_ratio=1.0; /* это число может меняется в различных
                              случаях */
             y=radius;
             delta=3-2*radius;
             for (x=0;x0) incx=1;
          else  if (delta_x==0) incx=0;
          else  incx= -1;

          if (delta_y>0) incy=1;
          else  if (delta_y==0) incy=0;
          else  incy= -1;

        /* Определение какое расстояние больше */
          delta_x=abs(delta_x);
          delta_y=abs(delta_y);
          if (delta_x>delta_y) distance=delta_x;
          else distance=delta_y;

        /* Вычерчивание линии */
          for (t=0; t<=distance+1; t++) {
             mempoint(startx,starty,color);
             xerr+=delta_x;
             yerr+=delta_y;
             if (xerr>distance) {
                xerr-=distance;
                startx+=incx;
             }
             if (yerr>distance) {
                yerr-=distance;
                starty+=incy;
             }
          }
       }



                                 "C" для профессиональных программистов
Глава IV                                                       -- 15 --


        /* Закрашивание прямоугольника  заданным цветом */

        void fill_box(startx,starty,endx,endy,color_code)
        int startx,starty,endx,endy,color_code;
        {
           register int i,begin,end;

           begin=startxendx ? startx:endx;

           for (i=begin;i<=end;++i)
              line(i,starty,i,endy,color_code);
         }


        /* Вычерчивание окружности с использованием алгоритма
           Брезенхама */
         void circle(x_center,y_center,radius,color_code)
         int x_center,y_center,radius,color_code;
         {
            register x,y,delta;
            asp_ratio=1.0; /* это число может меняется в различных
                              случаях */
             y=radius;
             delta=3-2*radius;
             for (x=0;x199 || y<0 || y>319) return;
            xor=color_code & 128;      /* проверка, устанавливался ли
                                          режим "исключающего ИЛИ" */
            color_code=color_code & 127;     /* маска старших битов */

            /*  Установка маски битов и битов режима цвета
                в правую позицию */

             bit_position=y%4; /* вычисление нужной позиции в байте */
             color_code<<=2*(3-bit_position); /* сдвиг кода цвета
                                        в нужную позицию */
             bit_mask.i>>=2*bit_position; /* сдвиг битовой маски в
                                          нужную позицию */

             /* определение требуемого байта в памяти терминала */



                                 "C" для профессиональных программистов
Глава IV                                                       -- 17 --


              index=x*40+(y%4);
              if (x%2) index+=8152; /* если нечетный, используется
                                       второй блок */

              /* Запись цвета */
               if (!xor) {  /* режим изменения цвета */
                  t=*(ptr+index) & bit_mask.c[0];
                  *(ptr+index)=t|color_code;
               }
               else {
                  t=*(ptr+index) | (char)0;
                  *(ptr+index)=t & color_code;
               }
            }










































                                 "C" для профессиональных программистов
Глава IV                                                       -- 18 --


               СОХРАНЕНИЕ И ЗАГРУЗКА ГРАФИЧЕСКИХ ИЗОБРАЖЕНИЙ
     -----------------------------------------------------------------

          Сохранение и   загрузка   графических  изображений  является
     довольно  простым  делом,  т.к.  образ  изображений  находится  в
     видеопамяти дисплея, а ее содержимое легко копировать на дисковый
     файл.   Главной   проблемой   является   необходимость   введения
     пользователем  имени  файла,  ввиду  того,  что  запрос о вводе и
     введенное имя файла сотрут часть информации на экране.  Для того,
     чтобы   избежать   этого,   разработаны   функции   save_pic()  и
     load_pic(),  тексты которых приводятся в данном  разделе.  Первая
     функция  сохраняет  14  верхних  строк  изображения,  чистит  эту
     область,  запрашивает имя файла и,  после  того,  как  оно  будет
     введено, восстанавливает изображение.


        /* сохранение  графического изображения */
        void save_pic()
        {
          char fname[80];
          FILE *fp;
          register int i,j;
          char far *ptr=(char far *) 0xB8000000; /* точка в
                                                 памяти   CGA  */
          char far *temp;
          unsigned char buf[14][80]; /* содержит образ экрана */

          temp=ptr;
        /* сохранение верхних строк текущего содержимого экрана */
          for (i=0;i<14;++i)
             for (j=0;j<80;++j) {
                buf[i][j]=*temp; /* четный байт */
                buf[i][j+1]=*(temp+8152); /* нечетный байт */
                *temp=0; *(temp+8152)=0; /* чистка позиций экрана */
                temp++;
             }
           goto_xy(0,0);
           printf("Имя файла:");
           gets(fname);
           if (!(fp=fopen(fname,"wb"))) {
               printf("Фвайл не может быть открыт\n");
               return;
           }

           temp=ptr;
       /* восстановление содержимого экрана */
          for (i=0;i<14;++i)
             for (j=0;j<80;++j) {
                *temp= buf[i][j]; /* четный байт */
                *(temp+8125)=buf[i][j+1]; /* нечетный байт */
                *temp=0; *(temp+8152)=0; /* чистка позиций экрана */
                temp++;
             }
        /* копирование изображения в файл  */


                                 "C" для профессиональных программистов
Глава IV                                                       -- 19 --


           for (i=0;i<8152;i++) {
              putc(*ptr,fp); /* четный байт */
              putc(*(ptr+8125),fp); /* нечетный байт */
              ptr++;
           }

           fclose(fp);
        }

       /* загрузка изображения */
         void load_pic()
         {
            char fname[80];
            FILE *fp;
            register int i,j;

            char far *ptr=(char far *) 0xB8000000; /* точка в
                                                   памяти   CGA  */
            char far *temp;
            unsigned char buf[14][80]; /* содержит образ экрана */

            temp=ptr;
        /* сохранение верхних строк текущего содержимого экрана */
            for (i=0;i<14;++i)
               for (j=0;j<80;j+=2) {
                  buf[i][j]=*temp;
                  buf[i][j+1]=*(temp+8152);
                  *temp=0; *(temp+8152)=0; /* чистка позиций экрана */
                  temp++;
               }
            goto_xy(0,0);
            printf("Имя файла:");
            gets(fname);
            if (!(fp=fopen(fname,"rb"))) {
               goto_xy(0,0);
               printf("Файл не может быть открыт\n");
               temp=ptr;
       /* восстановление содержимого экрана */
               for (i=0;i<14;++i)
                  for (j=0;j<80;j+=2) {
                     *temp= buf[i][j];
                     *(temp+8125)=buf[i][j+1];
                     temp++;
                  }
               return;
             }
      /* загрузка изображения из файла */
           for (i=0;i<8152;i++) {
              *ptr=getc(fp); /* четный байт */
              *(ptr+8125)=getc(fp); /* нечетный байт */
              ptr++;
           }

           fclose(fp);


                                 "C" для профессиональных программистов
Глава IV                                                       -- 20 --


        }

          Подпрограммы  начинают  обработку   видеопамяти,  начиная  с
     адреса,  содержащегося  в указателе temp  считывая  или записывая
     каждый четный и нечетный байты в порядке  возрастания их адресов.
     Такой   подход   позволяет   добиться   простоты   и  наглядности
     функционирования   рассмотренных   выше  подпрограмм.   В  случае
     обработки  видеопамяти  в  порядке  возрастания  адресов, вначале
     будет отображаться четная точка растра, а затем - нечетная.















































                                 "C" для профессиональных программистов
Глава IV                                                       -- 21 --


                         ДУБЛИРОВАНИЕ ЧАСТИ ЭКРАНА
     -----------------------------------------------------------------

          Иногда бывает  полезным  скопировать  часть  экрана в другую
     область.  Это легко выполнить  используя  функцию  copy(),  текст
     которой приводится ниже.

          /* копирование части экрана в другую область */
          void copy(startx,starty,endx,endy,x,y)
          int startx,starty;      /* верхняя левая координата */
          int endx,endy;          /* нижняя правая координата области
                                     копирования */
          int x,y;      /* верхняя левая координата области,
                           куда будет проводится копирование */
          {
             int i,j;
             unsigned char c;

             for (;startx                               і
     і  индекс        индекс                                         і
     і                                                               і
     і    і             0           1            2           3       і
     і    і                                                          і
     і    і                                                          і
     і    V                                                          і
     і                                                               і
     і    0          start_X1     start_Y1     end_X1     end_Y1     і
     і                                                               і
     і    1          start_X2     start_Y2     end_X2     end_Y2     і
     і                                                               і
     і    2          start_X3     start_Y3     end_X3      end_Y3    і
     і                                                               і
     і    3          start_X4     start_Y4     end_X4      end_Y4    і
     і    .             .                                            і
     і    .             .                                            і
     і    .             .                                            і
     і    n          start_Xn     start_Yn     end_Xn      end_Yn    і
     АДДДДДДДДДДДДДДДДДДДДДДДДДДДДДДДДДДДДДДДДДДДДДДДДДДДДДДДДДДДДДДДЩ

     Рис. 4-3. Условная организация массива.


          Определить объект  -  это  значит   разместить   в   массиве
     координаты  начальных  и  конечных  точек отрезков,  составляющих
     объект.  Например,  если объект представляет собой  прямоугольник
     вида:



                                 "C" для профессиональных программистов
Глава IV                                                       -- 26 --


                         0.0ЪДДДДДДДДДДДДДДДДДї0.10
                            і                 і
                            і                 і
                            і                 і
                            і                 і
                        10.0АДДДДДДДДДДДДДДДДДЩ10.10

     то в  массив,  определяющий   данный   прямоугольник,   заносятся
     следующие числа:

          object[0][0] = 0; object[0][1] = 0;
          object[0][0] = 0; object[0][3] = 10;

          object[1][0] = 0; object[1][1] = 10;
          object[1][0] = 10; object[1][3] = 10;

          object[2][0] = 10; object[2][1] = 10;
          object[2][0] = 10; object[2][3] = 0;

          object[3][0] = 10; object[3][1] = 0;
          object[3][0] = 0; object[3][3] = 0;

          После того,  как  объект  определен,  вы можете вращать его,
     используя функцию rotate_object(),  приведенную ниже,  по часовой
     стрелке  (клавиша  )  или  в  противоположную сторону (клавиша
     ).


          /* Вращение заданных объектов */

          void rotate_object(ob, theta, x, y, sides)
          double ob[][4]; /* описание объекта */
          double theta;   /* угол поворота в радианах */
          int x, y;
          int sides;
            {
            register int i, j;
            double tempx, tempy;
            char ch;

            for(;;)
              {
              ch = getch(); /* ввод признака направления вращения */
              switch(tolower(ch))
                {
                case 'l': /* вращение против часовой стрелки */
                  theta = theta < 0 ? -theta : theta;
                  break;
                case 'r': /* вращение по часовой стрелке  */
                  theta = theta > 0 ? -theta : theta;
                  break;
                default: return;
                }
              for(j=0; j<=sides; j++) /* стирание старых линий  */


                                 "C" для профессиональных программистов
Глава IV                                                       -- 27 --


                {
                line((int) ob[j][0], (int) ob[j][1],
                     (int) ob[j][2], (int) ob[j][3], 0);
                rotate_point(theta, &ob[j][0],
                             &ob[j][1], x, y);
                rotate_point(theta, &ob[j][2], &ob[j][3], x, y);
                line((int) ob[j][0], (int) ob[j][1],
                     (int) ob[j][2], (int) ob[j][3], 2);
                }
              }
            }

          Как показано  в описании параметров функции rotate_object(),
     вращение осуществляется вокруг центра, заданного координатами X и
     Y, на угол, величина которого задана параметром theta в радианах.
     Минимальное значение параметра theta равно 0.01  радиан. Заметим,
     что  объект  сначала  стирается  из старой области размещения,  а
     затем перерисовывается вновь.  Если это  условие  не  может  быть
     выполнено,  то  экран  окрашивается  в голубой цвет.  Необходимым
     условием   выполнения    программы    rotate_object()    является
     обязательное задание параметра sides.

          Приведенная ниже функция display_object() не имеет отношения
     к вращению объектов,  но она может  быть  полезна  при  работе  с
     объектами.  Она рисует на экране объекты,  определенные в массиве
     ob.

          /* отображение объекта на экране */
          void display_object(ob, sides)
          double ob[][4];
          int sides;
          {
            register int i;

            for(i=0; i0)
              incx=1;
            else
              if(delta_x==0)
                incx=0;
              else
                incx= -1;
            if(delta_y>0)
              incy=1;
            else
              if(delta_y==0)
                incy=0;
              else
                incy= -1;
            delta_x=abs(delta_x);
            delta_y=abs(delta_y);


                                 "C" для профессиональных программистов
Глава IV                                                       -- 30 --


            if(delta_x>delta_y)
              distance=delta_x;
            else
              distance=delta_y;

            /* рисунок отрезка */
            for(t=0; t<=distance; t++)
              {
              mempoint(start_x, start_y, color);
              x+=delta_x;
              y+=delta_y;
              if(x>distance)
                {
                x-=distance;
                start_x+=incx;
                }
              if(y>distance)
                {
                y-=distance;
                start_y+=incy;
                }
              }
            }

        /* запись точки в CGA/EGA */

         void mempoint(x,y,color_code)
         int x,y,color_code;
         {
            union mask {
               char c[2];
               int i;
            } bit_mask;
            int i,index,bit_position;
            unsigned char t;
            char xor; /* "НЕ-ИЛИ" цвета в случае его
                          изменения */
            char far *ptr=(char far *) 0xB8000000; /* точка в
                                                  памяти CGA */
            bit_mask.i=0xFF3F; /* 11111111 00111111 в
                                  двоичном виде */
            if (x<0 || x>199 || y<0 || y>319) return;
            xor=color_code & 128; /* проверка, устанавливался ли
                                     режим "НЕ-ИЛИ" */
            color_code=color_code & 127; /* маска старших битов */

            /*  установка битовой маски и битов режима цвета
                в правую позицию */

             bit_position=y%4; /* вычисление нужной позиции
                                  в байте */
             color_code<<=2*(3-bit_position); /* сдвиг кода цвета
                                        в нужную позицию */
             bit_mask.i>>=2*bit_position; /* сдвиг битовой маски в


                                 "C" для профессиональных программистов
Глава IV                                                       -- 31 --


                                          нужную позицию */

             /* определение требуемого байта в памяти терминала */

              index=x*40+(y%4);
              if (x%2) index+=8152; /* если нечетный, используется
                                       второй блок */

              /* запись цвета */

               if (!xor) {  /* режим изменения цвета */
                  t=*(ptr+index) & bit_mask.c[0];
                  *(ptr+index)=t|color_code;
               }
               else {
                  t=*(ptr+index) | (char)0;
                  *(ptr+index)=t & color_code;
               }
            }


        /* вращение точки вокруг центра с координатами
           в x_org и y_org, на угол theta */

          void rotate_point(theta,x,y,x_org,y_org)
          double theta,*x,*y;
          int x_org,y_org;
          {
             double tx,ty;

        /* нормализация X и Y к начальному адресу */
             tx=*x-x_org;
             ty=*y-y_org;

        /* вращение */
              *x=tx*cos(theta)-ty*sin(theta);
              *y=tx*sin(theta)-ty*cos(theta);

         /* возвращение значений координат */
              *x+=x_org;
              *y+=y_org;
           }


          /* Вращение заданных объектов */

          void rotate_object(ob, theta, x, y, sides)
          double ob[][4]; /* описание объекта */
          double theta;   /* угол поворота в радианах */
          int x, y;
          int sides;
            {
            register int i, j;
            double tempx, tempy;


                                 "C" для профессиональных программистов
Глава IV                                                       -- 32 --


            char ch;

            for(;;)
              {
              ch = getch(); /* ввод признака направления вращения */
              switch(tolower(ch))
                {
                case 'l': /* вращение против часовой стрелки */
                  theta = theta < 0 ? -theta : theta;
                  break;
                case 'r': /* вращение по часовой стрелке  */
                  theta = theta > 0 ? -theta : theta;
                  break;
                default: return;
                }
              for(j=0; j<=sides; j++) /* стирание старых линий  */
                {
                line((int) ob[j][0], (int) ob[j][1],
                     (int) ob[j][2], (int) ob[j][3], 0);
                rotate_point(theta, &ob[j][0],
                             &ob[j][1], x, y);
                rotate_point(theta, &ob[j][2], &ob[j][3], x, y);
                line((int) ob[j][0], (int) ob[j][1],
                     (int) ob[j][2], (int) ob[j][3], 2);
                }
              }
            }


          /* отображение объекта на экране */

          void display_object(ob, sides)
          double ob[][4];
          int sides;
            {
            register int i;

            for(i=0; i (),  <Страница вверх>
     (),  <Страница  вниз>  ()  и <Кон> () перемещают
     графический курсор в указанном  направлении  и  под  углом  в  45
     градусов.

          Для анализа  кодов операций чтения в программах используется
     функция bioskey().  Порядок подключения этой функции к  программе


                                 "C" для профессиональных программистов
Глава IV                                                       -- 37 --


     при компиляции описан в главе 1. В программу включены обращения к
     функциям, позволяющим вам рисовать и закрашивать прямоугольники и
     окружности,  рисовать линии,  копировать и перемещать изображение
     на экране,  сохранять на диске  и  загружать  с  него  содержимое
     экрана, отображать и вращать объекты.

          При изображении  линий,  прямоугольников  и  окружностей вам
     необходимо определить координаты двух точек.  Для прямоугольников
     -  это координаты двух противоположных углов.  Для линий задается
     начальная и конечная точки,  а для окружности - координаты центра
     и точки, через которую она будет проходить.

          Процесс выбора  этих точек выполняется путем нажатия клавиши
     <ВВОД> в момент,  когда графический курсор находится в  требуемой
     области.   Например,   для   изображения   линии  вы  перемещаете
     графический курсор в точку, где она должна начинаться и нажимаете
     клавишу  <ВВОД>  .  Затем  вы устанавливаете курсор в точку,  где
     линия заканчивается,  и  нажимаете  <ВВОД>  снова.  При   нажатии
     клавиши  <ВВОД>  выполняется загрузка переменных startx,  starty,
     endx и endy,  которые потом используются  в  качестве  параметров
     вызываемых  функций.  После  того,  как  координаты  точек  будут
     определены,  при нажатии клавиши <В> рисуется квадрат,  а    -
     квадрат закрашивается,   при  нажатии    рисуется  линия,  при
     нажатии   <С>   рисуется   окружность,   а   <Н>   -   окружность
     закрашивается.

          Для копирования  или  перемещения  части  экрана  вы  должны
     определить верхний левый и нижний правый углы области, которую вы
     хотите   переместить   (нажатием   клавиши  <ВВОД>  ).  Затем  вы
     перемещаете курсор в верхний левый угол области,  куда вы  хотите
     переместить  изображение.  Для  пересылки  изображения  требуется
     нажать клавишу <М>,  а для  копирования  -  <Х>.  Запомните,  что
     старое  изображение  в области,  куда осуществляется копирование,
     будет уничтожено.

          Для вращения объекта вам необходимо определить  сам  объект,
     путем нажатия клавиши .  Затем,  используя клавишу <ВВОД>,  вы
     должны определить  начальные  и  конечные  координаты  точек  для
     отрезков по периметру выбранного объекта.  Процесс выбора объекта
     вращения  и   определения   его   границ   реализуется   функцией
     define_object().   Вращение   объекта  начинается  после  нажатия
     клавиши <А>.  Для определения направления  вращения  используются
     клавиши     (по  часовой  стрелке)  или    (против  часовой
     стрелки).  Остановить  процесс  вращения  можно  нажатием   любой
     клавиши, кроме  или <А>.

          Для остановки работы программы используется клавиша . При
     желании вы можете включить  в  программу  функции  для  работы  с
     "мышью". Пример выходных данных программы показан на рисунке 4-5.

     _________________________________________________________________

     Рисунок 4-5 на стр. 163 не может быть воспроизведен имеющимися
     средствами. (Ред. пер. И.Бычковский)


                                 "C" для профессиональных программистов
Глава IV                                                       -- 38 --


     _________________________________________________________________

     Рис. 4-5. Простейшие результаты работы программы рисования.


          А теперь приведем всю программу рисования целиком.


     /* Программа    для    CGA/EGA,   позволяющая   рисовать   линии,
     прямоугольники и  окружности.  Вы  можете  нарисовать  какой-либо
     объект  и  вращать его по часовой или против часовой стрелки.  Вы
     так же  можете  копировать  графическое  изображение  на  диск  и
     загружать его с диска. */

          #define NUM_SIDES 20 /* число сторон объекта;
                                  при необходимости увеличивается */
          #include "dos.h"
          #include "stdio.h"
          #include "math.h"

          void mode(), line(), box(), fill_box();
          void mempoint(), palette(), xhairs();
          void circle(), plot_circle(), fill_circle();
          void rotate_point(), rotate_object(), goto_xy();
          void display_object(), copy(), move();
          void save_pic(), load_pic();
          unsigned char read_point();

          /*  Этот массив содержит динамически меняющиеся
              координаты объекта.
          */
          double object[NUM_SIDES][4];
          double asp_ratio; /* содержит коэффициент сжатия для
                               окружностей */

          main()
            {
            union k{
              char c[2];
              int i;
               } key ;

            int x=10, y=10; /* текущая позиция экрана */
            int cc=2;   /* текущий цвет */
            int on_flag=1; /* признак использования карандаша */
            int pal_num=1; /* номер палитры */

            int startx=0, starty=0, endx=0, endy=0;
            int first_point=1;
            int inc=1; /* шаг пересылки */
            int sides=0; /* количество сторoн выбранного объекта */
            int i;

            mode(4); /* переключатель режима CGA/EGA */


                                 "C" для профессиональных программистов
Глава IV                                                       -- 39 --


            palette(0); /* палитра 0 */

            xhairs(x, y); /* указатель курсора */
             do
              {
              key.i = bioskey(0);
              xhairs(x, y);
              if(!key.c[0]) switch(key.c[1])
                {
                case 75: /* влево */
                  if(on_flag) line(x, y, x, y-inc, cc);
                  y -= inc;
                  break;
                case 77: /* вправо */
                  if(on_flag) line(x, y, x, y+inc, cc);
                  y += inc;
                  break;
                case 72: /* вверх */
                    if(on_flag) line(x, y, x-inc, y, cc);
                  x -= inc;
                  break;
                case 80: /* вниз */
                    if(on_flag) line(x, y, x+inc, y, cc);
                  x += inc;
                  break;
                case 71: /* вверх и влево */
                  if(on_flag) line(x, y, x-inc, y-inc, cc);
                  x -= inc;
                  y -= inc;
                  break;
                case 73: /* вверх и вправо */
                  if(on_flag) line(x, y, x-inc, y+inc, cc);
                  x -= inc;
                  y += inc;
                  break;
                case 79: /* вниз и влево */
                  if(on_flag) line(x, y, x+inc, y-inc, cc);
                  x += inc;
                  y -= inc;
                  break;
                case 81: /* вниз и вправо */
                  if(on_flag) line(x, y, x+inc, y+inc, cc);
                  x += inc;
                  y += inc;
                  break;
                case 59: /* F1 - медленно */
                  inc=1;
                  break;
                case 60: /* F2 - быстро   */
                  inc=5;
                  break;
                }
              else switch(tolower(key.c[0]))
                {


                                 "C" для профессиональных программистов
Глава IV                                                       -- 40 --


                case 'o': /* переключение шаблона */
                  on_flag = !on_flag;
                  break;
                case '1': cc=1; /* цвет 1 */
                  break;
                case '2': cc=2; /* цвет 2 */
                  break;
                 case '3': cc=3; /* цвет 3 */
                  break;
                case '0': cc=0; /* цвет 0 */
                  break;
                case 'b':
                  box(startx, starty, endx, endy, cc);      break;
                case 'f':
                  fill_box(startx, starty, endx, endy, cc); break;
                case 'l':
                  line(startx, starty, endx, endy, cc);     break;
                case 'c':
                  circle(startx, starty, endy-starty, cc);  break;
                case 'h':
                  fill_circle(startx, starty, endy-starty, cc); break;
                case 's':
                  save_pic();                               break;
                case 'r':
                  load_pic();                               break;
                case 'm': /* пересылка фрагмента */
                  move(startx, starty, endx, endy, x, y);   break;
                case 'x': /* копирование фрагмента */
                  copy(startx, starty, endx, endy, x, y);   break;
                case 'd': /* определить объект вращения */
                  sides = define_objekt(object, x, y);      break;
                case 'a': /* вращение объекта */
                  rotate_objekt(object, 0.05, x, y, sides); break;
                case '\r': /* набор конечных точек для линий, кругов
                              или прямоугольников */
                  if(first_point)
                    { startx = x, starty = y; }
                  else
                    { endx = x, endy = y; }
                  first_point = !first_point;               break;
                case 'p':
                  pal_num = pal_num==1 ? 2:1;
                  palette(pal_num);
                }
              xhairs(x, y);
              }
            while (key.c[0]!='q');
            getchar();
            mode(2);
            }

          /* установка палитры */
          void palette(pnum)
          int pnum;


                                 "C" для профессиональных программистов
Глава IV                                                       -- 41 --


            {
            union REGS r;
            r.h.bh = 1; /* код 4 режима графики */
            r.h.bl = pnum;
            r.h.ah = 11; /* установка палитры */
            int86(0x10, &r, &r);
            }

          /* установка видео-режима */
          void mode (mode_code)
          int mode_code;
            {
            union REGS r;

            r.h.al = mode_code;
            r.h.ah = 0;
            int86(0x10,&r, &r);
            }

          /* изображение прямоугольника */
          void box(sx, sy, ex, ey, c)
          int sx, sy, ex, ey, c;
            {
            line(sx, sy, ex, sy, c);
            line(sx, sy, sx, ey, c);
            line(sx, ey, ex, ey, c);
            line(ex, sy, ex, ey, c);
            }

      /* изображение линии заданного цвета с использованием
         алгоритма Брезенхама */
       void line(startx,starty,endx,endy,color)
       int startx,starty,endx,endy,color;
       {
         register int t,distance;
         int xerr=0,yerr=0,delta_x,delta_y;
         int incx,incy;

       /* вычисление расстояния в обоих направлениях         */
         delta_x=endx-startx;
         delta_y=endy-starty;

       /* определение направления шага,
          шаг вычисляется либо по вертикальной, либо горизонтальной
          линии                   */
          if (delta_x>0) incx=1;
          else  if (delta_x==0) incx=0;
          else  incx= -1;

          if (delta_y>0) incy=1;
          else  if (delta_y==0) incy=0;
          else  incy= -1;

        /* определение какое расстояние больше */


                                 "C" для профессиональных программистов
Глава IV                                                       -- 42 --


          delta_x=abs(delta_x);
          delta_y=abs(delta_y);
          if (delta_x>delta_y) distance=delta_x;
          else distance=delta_y;

        /* изображение линии */
          for (t=0; t<=distance+1; t++) {
             mempoint(startx,starty,color);
             xerr+=delta_x;
             yerr+=delta_y;
             if (xerr>distance) {
                xerr-=distance;
                startx+=incx;
             }
             if (yerr>distance) {
                yerr-=distance;
                starty+=incy;
             }
          }
       }


        /* закрашивание прямоугольника в заданный цвет */

        void fill_box(startx,starty,endx,endy,color_code)
        int startx,starty,endx,endy,color_code;
        {
           register int i,begin,end;

           begin=startxendx ? startx:endx;

           for (i=begin;i<=end;++i)
              line(i,starty,i,endy,color_code);
         }


        /* изображение   окружности   с    использованием    алгоритма
           Брезенхама */

         void circle(x_center,y_center,radius,color_code)
         int x_center,y_center,radius,color_code;
         {
            register x,y,delta;
            asp_ratio=1.0; /* это число меняется в различных
                              случаях */
             y=radius;
             delta=3-2*radius;
             for (x=0;x199 || y<0 || y>319) return;
            xor=color_code & 128; /* проверка, устанавливался ли
                                     режим "исключающего ИЛИ" */
            color_code=color_code & 127; /* маска старших битов */

            /*  установка битовой маски и битов режима цвета
                в правую позицию */

             bit_position=y%4; /* вычисление нужной позиции
                                  в байте */
             color_code<<=2*(3-bit_position); /* сдвиг кода цвета
                                        в нужную позицию */
             bit_mask.i>>=2*bit_position; /* сдвиг битовой маски в
                                          нужную позицию */

             /* определение требуемого байта в памяти терминала */

              index=x*40+(y%4);
              if (x%2) index+=8152; /* если нечетный, используется
                                       второй блок */

              /* запись цвета */

               if (!xor) {  /* режим изменения цвета */
                  t=*(ptr+index) & bit_mask.c[0];
                  *(ptr+index)=t|color_code;
               }
               else {
                  t=*(ptr+index) | (char)0;
                  *(ptr+index)=t & color_code;
               }
            }

        /* отображение графического курсора */
         void xhairs(x,y)
         int x,y;
         {
            line(x-4,y,x+3,y,1|128);
            line(x,y+4,x,y-3,1|128);

          }

        /* чтение байта из оперативной памяти CGA/EGA */
         unsigned char read_point(x,y)
         int x,y;
         {


                                 "C" для профессиональных программистов
Глава IV                                                       -- 45 --


            union mask {
               char c[2];
               int i;
            } bit_mask;
            int i,index,bit_position;
            unsigned char t;
            char xor; /* "исключающее ИЛИ" цвета в случае его
                          изменения */
            char far *ptr=(char far *) 0xB8000000; /* точка в
                                                  памяти CGA */
            bit_mask.i=3; /* 11111111 00111111 в
                                  двоичном виде */
            if (x<0 || x>199 || y<0 || y>319) return 0;

            /*  установка битовой маски и битов режима цвета
                в правую позицию */
             bit_position=y%4; /* вычисление нужной позиции
                                  в байте */
             bit_mask.i<<=2*(3-bit_position);

             /* определение требуемого байта в памяти терминала */
              index=x*40+(y>>4);
              if (x%2) index+=8152; /* если нечетный, используется
                                       второй блок */

              /* запись цвета */
               t=*(ptr+index) & bit_mask.c[0];
               t>>=2*(3-bit_position);
               return t;
            }

       /* сохранение  графического изображения  */

        void save_pic()
        {
          char fname[80];
          FILE *fp;
          register int i,j;
          char far *ptr=(char far *) 0xB8000000; /* точка в
                                                 памяти   CGA  */
          char far *temp;
          unsigned char buf[14][80]; /* содержит образ экрана */

          temp=ptr;
        /* сохранение верхних строк текущего содержимого экрана */
          for (i=0;i<14;++i)
             for (j=0;j<80;++j) {
                buf[i][j]=*temp; /* четный байт */
                buf[i][j+1]=*(temp+8152); /* нечетный байт */
                *temp=0; *(temp+8152)=0; /* чистка позиций экрана */
                temp++;
             }
           goto_xy(0,0);
           printf("Имя файла:");


                                 "C" для профессиональных программистов
Глава IV                                                       -- 46 --


           gets(fname);
           if (!(fp=fopen(fname,"wb"))) {
               printf("Файл не может быть открыт\n");
               return;
           }

           temp=ptr;
       /* восстановление содержимого экрана */
          for (i=0;i<14;++i)
             for (j=0;j<80;++j) {
                *temp= buf[i][j]; /* четный байт */
                *(temp+8125)=buf[i][j+1]; /* нечетный байт */
                *temp=0; *(temp+8152)=0; /* чистка позиций экрана */
                temp++;
             }
        /* копирование изображения в файл  */
           for (i=0;i<8152;i++) {
              putc(*ptr,fp); /* четный байт */
              putc(*(ptr+8125),fp); /* нечетный байт */
              ptr++;
           }

           fclose(fp);
        }


       /* загрузка изображения */

         void load_pic()
         {
            char fname[80];
            FILE *fp;
            register int i,j;

            char far *ptr=(char far *) 0xB8000000; /* точка в
                                                   памяти   CGA  */
            char far *temp;
            unsigned char buf[14][80]; /* содержит образ экрана */

            temp=ptr;
        /* сохранение верхних строк текущего содержимого экрана */
            for (i=0;i<14;++i)
               for (j=0;j<80;j+=2) {
                  buf[i][j]=*temp;
                  buf[i][j+1]=*(temp+8152);
                  *temp=0; *(temp+8152)=0; /* чистка позиций экрана */
                  temp++;
               }
            goto_xy(0,0);
            printf("Имя файла:");
            gets(fname);
            if (!(fp=fopen(fname,"rb"))) {
               goto_xy(0,0);
               printf("Файл не может быть открыт\n");


                                 "C" для профессиональных программистов
Глава IV                                                       -- 47 --


               temp=ptr;
       /* восстановление содержимого экрана */
               for (i=0;i<14;++i)
                  for (j=0;j<80;j+=2) {
                     *temp= buf[i][j];
                     *(temp+8125)=buf[i][j+1];
                     temp++;
                  }
               return;
             }
      /* загрузка изображения из файла */
           for (i=0;i<8152;i++) {
              *ptr=getc(fp); /* четный байт */
              *(ptr+8125)=getc(fp); /* нечетный байт */
              ptr++;
           }

           fclose(fp);
        }

       /*  поместить курсор в заданное положение */
           void goto_xy(x,y)
           int x,y;
           {
              r.h.ah=2; /* адресация курсора */
              r.h.dl=y; /* координата столбца */
              r.h.dh=x; /* координата строки */
              r.h.bh=0; /* видеостраница */
              int86(0x10,&r,&r);
            }


       /* копирование части экрана в другую область */

          void copy(startx,starty,endx,endy,x,y)
          int startx,starty; /* верхняя левая координата */
          int endx,endy; /* нижняя правая координата области
                            копирования */
          int x,y; /* верхняя левая координата области,
                     куда будет проводится копирование */
          {
             int i,j;
             unsigned char c;

             for (;startx 0 ? -theta : theta;
                  break;
                default: return;
                }
              for(j=0; j<=sides; j++) /* стирание старых линий  */
                {
                line((int) ob[j][0], (int) ob[j][1],
                     (int) ob[j][2], (int) ob[j][3], 0);
                rotate_point(theta, &ob[j][0], &ob[j][1], x, y);
                rotate_point(theta, &ob[j][2], &ob[j][3], x, y);
                line((int) ob[j][0], (int) ob[j][1],
                     (int) ob[j][2], (int) ob[j][3], 2);
                }
              }
            }


          /* отображение объекта на экране */
          void display_object(ob, sides)
          double ob[][4];
          int sides;
            {
            register int i;

            for(i=0; i




?????? ???????????