mirror of
https://github.com/KolibriOS/kolibrios.git
synced 2024-12-22 22:56:53 +03:00
83bfd951f0
git-svn-id: svn://kolibrios.org@3001 a494cfbc-eb01-0410-851d-a64ba20cac60
1252 lines
27 KiB
NASM
1252 lines
27 KiB
NASM
format MS COFF
|
||
public EXPORTS
|
||
section '.flat' code readable align 16
|
||
|
||
include 'vectors.inc' ;vectors functions constant
|
||
|
||
macro swap v1, v2 {
|
||
push v1
|
||
push v2
|
||
pop v1
|
||
pop v2
|
||
}
|
||
|
||
BUF_STRUCT_SIZE equ 17 ;размер структуры, описывающей буфер
|
||
BUF_MAX_COUNT equ 8 ;максимальное число буферов
|
||
|
||
fun_draw_pixel dd drawpixel_scrn ;указатель на функцию рисования точки
|
||
active_buffer: ;начало структуры активного буфера (буфера в который делается рисование фигур или текста)
|
||
dd 0 ;указатель на буфер изображения
|
||
dw 0 ; +4 left
|
||
dw 0 ; +6 top
|
||
dw 0 ; +8 w ширина буфера
|
||
dw 0 ;+10 h высота буфера
|
||
dd 0 ;+12 color цвет фона
|
||
db 24 ;+16 bit in pixel глубина цвета (в данной версии еще не используется)
|
||
rb BUF_STRUCT_SIZE * BUF_MAX_COUNT ;резервируем место для структур, описывающих буфера
|
||
;в активном буфере будет содержаться точная копия одной из этих структур
|
||
|
||
|
||
active_buffer_left equ active_buffer+ 4
|
||
active_buffer_top equ active_buffer+ 6
|
||
active_buffer_w equ active_buffer+ 8
|
||
active_buffer_h equ active_buffer+10
|
||
active_buffer_color equ active_buffer+12
|
||
|
||
|
||
;-----------------------------------------------------------------------------
|
||
;функция для выделения памяти
|
||
;input:
|
||
; ecx = size data
|
||
;otput:
|
||
; eax = pointer to memory
|
||
mem_Alloc:
|
||
push ebx
|
||
mov eax,68
|
||
mov ebx,12
|
||
int 0x40
|
||
pop ebx
|
||
ret
|
||
;-----------------------------------------------------------------------------
|
||
;функция для освобождения памяти
|
||
;input:
|
||
; ecx = pointer to memory
|
||
mem_Free:
|
||
push eax ebx
|
||
cmp ecx,0
|
||
jz @f
|
||
mov eax,68
|
||
mov ebx,13
|
||
int 0x40
|
||
@@:
|
||
pop ebx eax
|
||
ret
|
||
|
||
;функция рисующая точку сразу на экран (без участия буфера)
|
||
align 4
|
||
drawpixel_scrn:
|
||
bt bx,15
|
||
jc @f
|
||
bt cx,15
|
||
jc @f
|
||
; cmp bx,300
|
||
; jge @f
|
||
; cmp cx,300
|
||
; jge @f
|
||
int 0x40
|
||
@@:
|
||
ret
|
||
|
||
;функция рисующая точку в активном буфере
|
||
align 4
|
||
drawpixel_buf:
|
||
bt bx,15 ;проверяем знак числа, если координата меньше 0
|
||
jc @f ;тогда точка в экран не попала, конец функции
|
||
bt cx,15
|
||
jc @f
|
||
cmp bx,word[active_buffer_w] ;проверяем координату точки, если больше ширины буфера
|
||
jge @f ;тогда точка в экран не попала, конец функции
|
||
cmp cx,word[active_buffer_h]
|
||
jge @f
|
||
|
||
push esi
|
||
xor esi,esi ;тут будет указатель на пиксель из активного буфера
|
||
mov si,word[active_buffer_w] ;ширина буфера по оси x
|
||
imul esi,ecx ;size_x*y
|
||
add esi,ebx ;size_x*y+x
|
||
lea esi,[esi+esi*2] ;(size_x*y+x)*3
|
||
add esi,dword[active_buffer] ;ptr+(size_x*y+x)*3
|
||
|
||
mov word[esi],dx ;копируем зеленый и синий спектр
|
||
ror edx,16 ;крутим цвет на 2 байта
|
||
mov byte[esi+2],dl ;копируем красный спектр
|
||
ror edx,16 ;крутим цвет назад
|
||
pop esi
|
||
@@:
|
||
ret
|
||
|
||
;функция создающая новый буфер, принимает параметры нужные для структуры
|
||
;input:
|
||
; [esp+8] = bit on pixel, index created buffer
|
||
; [esp+10] = color
|
||
; [esp+14] = size: w,h
|
||
; [esp+18] = size: l,t
|
||
align 4
|
||
buf_create:
|
||
push ebp
|
||
mov ebp,esp
|
||
cmp byte[ebp+8],1 ;проверка правильности индекса создаваемого буфера
|
||
jl .error_ind ;пользователь указал индекс меньше 1-цы, ... error :(
|
||
cmp byte[ebp+8],BUF_MAX_COUNT ;проверка правильности индекса создаваемого буфера
|
||
jg .error_ind ;пользователь указал индекс больше максимально возможного, ... error :(
|
||
push eax ecx edi esi
|
||
mov eax,dword[ebp+14] ;берем ширину и высоту буфера
|
||
ror eax,16 ;меняем местами ширину и высоту, так будет удобнее при выводе для функции 7
|
||
mov dword[active_buffer_w],eax ;помещаем значения в структуру активного буфера
|
||
|
||
xor ecx,ecx ;тут вычисляем сколько памяти нужно для этого буфера
|
||
mov cx,ax ;берем нижний размер буфера
|
||
shr eax,16 ;затираем нижний размер, в eax теперь только верхний размер, на месте нижнего
|
||
imul ecx,eax ;умножаем высоту на ширину (может и наоборот ширину на высоту)
|
||
imul ecx,3 ; 24 bit = 3, 32 bit = 4 ... work if only 24
|
||
call mem_Alloc ;просим память из системы
|
||
mov dword[active_buffer],eax ;копируем указатель на полученую память в структуру активного буфера
|
||
|
||
mov eax,dword[ebp+18] ;берем отступы слева и справа
|
||
ror eax,16
|
||
mov dword[active_buffer_left],eax
|
||
|
||
mov eax,dword[ebp+10] ;get color - берем цвет для фона
|
||
mov dword[active_buffer_color],eax
|
||
|
||
;копируем всю структуру активного буфера, в память отведенную для буферов
|
||
;иначе если появится новый буфер, то он затрет собой активную структуру
|
||
;потому нужно дублировать эту информацию
|
||
mov di,word[ebp+8] ;copy buffer struct
|
||
and edi,0xff ;index <= 255
|
||
mov ecx,BUF_STRUCT_SIZE ;в ecx размер копируемых данных
|
||
imul edi,ecx
|
||
mov esi,active_buffer
|
||
add edi,esi
|
||
rep movsb ;повторяем копирование по байту, пока ecx станет не равно 0
|
||
|
||
push word[ebp+8] ;при создании буфера, в нем может быть мусор,
|
||
call buf_clear ;потому чистим его фоновым цветом
|
||
pop esi edi ecx eax
|
||
.error_ind:
|
||
pop ebp
|
||
ret 14
|
||
|
||
;функция установки активного буфера, если на входе 0 - то включается режим рисования на экран без буфера
|
||
;input:
|
||
; [esp+8] = index buffer (0-screen)
|
||
align 4
|
||
set_active_buf:
|
||
push ebp
|
||
mov ebp,esp
|
||
|
||
cmp word[ebp+8],0
|
||
jne @f
|
||
.to_scrn:
|
||
mov dword[fun_draw_pixel],drawpixel_scrn ;рисование в экран
|
||
jmp .end_fun
|
||
@@:
|
||
cmp byte[ebp+8],BUF_MAX_COUNT ;if buffer index out of range
|
||
jg .to_scrn
|
||
mov dword[fun_draw_pixel],drawpixel_buf ;рисование в буфер
|
||
push ecx esi edi
|
||
mov si,word[ebp+8] ;copy buffer struct
|
||
and esi,0xff ;index <= 255
|
||
mov ecx,BUF_STRUCT_SIZE
|
||
imul esi,ecx
|
||
mov edi,active_buffer
|
||
add esi,edi
|
||
rep movsb
|
||
pop edi esi ecx
|
||
cmp dword[active_buffer],0 ;if buffer is empty
|
||
je .to_scrn
|
||
.end_fun:
|
||
pop ebp
|
||
ret 2
|
||
|
||
;функция очистки буфера фоновым цветом
|
||
;input:
|
||
; [esp+8] = index buffer (0-screen)
|
||
align 4
|
||
buf_clear:
|
||
push ebp
|
||
mov ebp,esp
|
||
push eax ebx ecx edi
|
||
|
||
mov di,word[ebp+8] ;get pointer to buffer struct
|
||
and edi,0xff ;index <= 255
|
||
imul edi,BUF_STRUCT_SIZE
|
||
add edi,active_buffer ;edi = pointer to buffer struct
|
||
|
||
cmp dword[edi],0 ;проверяем пустой указатель на буфер или нет
|
||
je .no_draw ;если пустой, то выход
|
||
xor ecx,ecx ;тут будет размер буфера в пикселях
|
||
mov cx,word[edi+8] ;active_buffer_w]
|
||
xor eax,eax
|
||
mov ax,word[edi+10] ;active_buffer_h]
|
||
imul ecx,eax ;ecx=x*y
|
||
mov ebx,dword[edi+12] ;active_buffer_color]
|
||
mov ax,bx
|
||
shr ebx,16
|
||
;imul ecx,3
|
||
;rep stosb
|
||
push dword[edi] ;save value in pointer
|
||
pop edi ;get value in pointer
|
||
@@:
|
||
mov word[edi],ax
|
||
add edi,2
|
||
mov byte[edi],bl
|
||
inc edi
|
||
loop @b
|
||
.no_draw:
|
||
pop edi ecx ebx eax
|
||
pop ebp
|
||
ret 2
|
||
|
||
;функция рисующая содержимое буфера на экране, использует КОС функцию номер 7
|
||
;input:
|
||
; [esp+8] = index buffer (0-screen)
|
||
align 4
|
||
draw_buf:
|
||
push ebp
|
||
mov ebp,esp
|
||
|
||
mov di,word[ebp+8] ;get pointer to buffer struct
|
||
and edi,0xff ;index <= 255
|
||
imul edi,BUF_STRUCT_SIZE
|
||
add edi,active_buffer ;edi = pointer to buffer struct
|
||
|
||
mov eax,7
|
||
mov ebx,dword[edi] ;active_buffer]
|
||
mov ecx,dword[edi+8] ;active_buffer_w] ;ecx = w*0xffff+h
|
||
ror ecx,16
|
||
|
||
;push word[edi+4] ;active_buffer_left] ;загрузка точки левого верхнего угла
|
||
;pop dx
|
||
mov dx,word[edi+4]
|
||
shl edx,16
|
||
;push word[edi+6] ;active_buffer_top] ;загрузка точки левого верхнего угла
|
||
;pop dx
|
||
mov dx,word[edi+6]
|
||
int 0x40
|
||
|
||
pop ebp
|
||
ret 2
|
||
|
||
;функция очищающая память, занимаемую буфером
|
||
;input:
|
||
; [esp+8] = index buffer (0-screen)
|
||
align 4
|
||
buf_delete:
|
||
push ebp
|
||
mov ebp,esp
|
||
|
||
mov cx,word[ebp+8] ;get pointer to buffer struct
|
||
and ecx,0xff ;index <= 255
|
||
imul ecx,BUF_STRUCT_SIZE
|
||
add ecx,active_buffer ;edi = pointer to buffer struct
|
||
|
||
push dword[ecx] ;save value in pointer
|
||
pop ecx ;get value in pointer
|
||
call mem_Free
|
||
|
||
pop ebp
|
||
ret 2
|
||
|
||
|
||
;функция рисующая линию
|
||
;input:
|
||
; [esp+8] = p0
|
||
; [esp+12] = p1
|
||
; [esp+16] = color
|
||
loc_0 equ byte[ebp-4]
|
||
loc_1 equ word[ebp-6]
|
||
loc_2 equ word[ebp-8]
|
||
align 4
|
||
line_brs:
|
||
push ebp
|
||
mov ebp,esp
|
||
sub esp,6 ;=1+2*2
|
||
pushad ;eax ebx ecx edx si di
|
||
mov edx,dword[ebp+16]
|
||
;---
|
||
mov ax,word[ebp+14] ;y1
|
||
; cmp ax,0 ;if y1<0 return
|
||
; jl .coord_end
|
||
; cmp word[ebp+10],0 ;if y0<0 return
|
||
; jl .coord_end
|
||
sub ax,word[ebp+10] ;y1-y0
|
||
bt ax,15
|
||
jae @f
|
||
neg ax
|
||
inc ax
|
||
@@:
|
||
mov bx,word[ebp+12] ;x1
|
||
; cmp bx,0 ;if x1<0 return
|
||
; jl .coord_end
|
||
; cmp word[ebp+8],0 ;if x0<0 return
|
||
; jl .coord_end
|
||
sub bx,word[ebp+8] ;x1-x0
|
||
bt bx,15
|
||
jae @f
|
||
neg bx
|
||
inc bx
|
||
@@:
|
||
|
||
mov byte[ebp-4],byte 0 ;bool steep=false
|
||
cmp ax,bx
|
||
jle @f
|
||
mov byte[ebp-4],byte 1 ;bool steep=true
|
||
swap word[ebp+8],word[ebp+10] ;swap(x0, y0);
|
||
swap word[ebp+12],word[ebp+14] ;swap(x1, y1);
|
||
@@:
|
||
mov ax,word[ebp+8] ;x0
|
||
cmp ax,word[ebp+12] ;if(x0>x1)
|
||
jle @f
|
||
swap word[ebp+8],word[ebp+12] ;swap(x0, x1);
|
||
swap word[ebp+10],word[ebp+14] ;swap(y0, y1);
|
||
@@:
|
||
|
||
; int deltax si
|
||
; int deltay di
|
||
; int error ebp-6
|
||
; int ystep ebp-8
|
||
|
||
mov ax,word[ebp+8] ;x=x0
|
||
mov si,word[ebp+12] ;x1
|
||
sub si,ax ;deltax = x1-x0
|
||
mov bx,si
|
||
shr bx,1
|
||
mov loc_1,bx ;error = deltax/2
|
||
|
||
mov ax,word[ebp+10] ;y=y0
|
||
mov di,word[ebp+14] ;y1
|
||
mov loc_2,word -1 ;ystep = -1
|
||
cmp ax,di ;if (y0<y1) ystep = 1;
|
||
jge @f
|
||
mov loc_2,word 1 ;ystep = 1
|
||
@@:
|
||
sub di,ax ;y1-y0
|
||
|
||
bts di,15
|
||
jae @f
|
||
neg di
|
||
inc di
|
||
@@:
|
||
and di,0x7fff ;deltay = abs(y1-y0)
|
||
|
||
mov eax,1 ;function, draw point
|
||
xor ebx,ebx
|
||
xor ecx,ecx
|
||
|
||
cmp byte[ebp-4],0
|
||
jne .coord_yx
|
||
mov bx,word[ebp+10] ;y0
|
||
mov cx,word[ebp+8] ;x0
|
||
|
||
@@: ;for (x=x0 ; x<x1; x++) ;------------------------------------
|
||
cmp cx,word[ebp+12]
|
||
jg @f ;jge ???
|
||
call dword[fun_draw_pixel]
|
||
|
||
sub loc_1,di ;error -= deltay
|
||
cmp loc_1,0 ;if(error<0)
|
||
jge .if0
|
||
add bx,loc_2 ;y += ystep
|
||
add loc_1,si ;error += deltax
|
||
.if0:
|
||
inc cx
|
||
jmp @b
|
||
@@:
|
||
|
||
jmp .coord_end
|
||
.coord_yx:
|
||
mov bx,word[ebp+8] ;x0
|
||
mov cx,word[ebp+10] ;y0
|
||
|
||
@@: ;for (x=x0 ; x<x1; x++) ;------------------------------------
|
||
cmp bx,word[ebp+12]
|
||
jg @f ;jge ???
|
||
call dword[fun_draw_pixel]
|
||
|
||
sub loc_1,di ;error -= deltay
|
||
cmp loc_1,0 ;if(error<0)
|
||
jge .if1
|
||
add cx,loc_2 ;y += ystep
|
||
add loc_1,si ;error += deltax
|
||
.if1:
|
||
inc bx
|
||
jmp @b
|
||
@@:
|
||
|
||
.coord_end:
|
||
;---
|
||
popad
|
||
mov esp,ebp ; восстанавливаем стек
|
||
pop ebp
|
||
ret 12
|
||
|
||
|
||
;input:
|
||
; [esp+8] = p0
|
||
; [esp+12] = p1
|
||
; [esp+16] = p2
|
||
; [esp+20] = color
|
||
align 4
|
||
cruve_bezier:
|
||
push ebp
|
||
mov ebp,esp
|
||
|
||
pushad
|
||
|
||
;float t, xt,yt;
|
||
;for(t=.0;t<1.;t+=.005){
|
||
; xt=pow(1.-t,2)*x0+2*t*(1.-t)*x1+pow(t,2)*x2;
|
||
; yt=pow(1.-t,2)*y0+2*t*(1.-t)*y1+pow(t,2)*y2;
|
||
; dc.SetPixel(xt,yt,255L);
|
||
;}
|
||
.beg_fun: ;для входа из другой функции
|
||
|
||
|
||
mov edx,dword[ebp+20] ;set cruve color
|
||
xor ebx,ebx
|
||
xor ecx,ecx
|
||
|
||
finit
|
||
|
||
; calculate delta t - вычисление шага изменения параметра t для рисования кривой Безье
|
||
push dword[ebp+8]
|
||
push dword[ebp+12]
|
||
call line_len4i ;определяем длину отрезка p0p1
|
||
fld dword[o_len]
|
||
push dword[ebp+12]
|
||
push dword[ebp+16]
|
||
call line_len4i ;определяем длину отрезка p1p2
|
||
fadd dword[o_len] ;находим сумарную длину (p0p1 + p1p2)
|
||
fadd st0,st0 ; умножаем длинну (p0p1 + p1p2) на 2
|
||
ftst
|
||
fstsw ax
|
||
|
||
fld1
|
||
sahf
|
||
jle @f ;избегаем деления на 0
|
||
fdiv st0,st1 ;находим шаг для изменения параметра t по формуле 1 / (2 * (p0p1 + p1p2))
|
||
; т.к. прямая в некоторых случаях "рвется", то я думаю что данная формула не оптимальна,
|
||
; но ничего лучшего я пока не придумал, ... :(
|
||
@@:
|
||
fstp dword[delt_t]
|
||
|
||
finit
|
||
|
||
;fild word[ebp+18] ;y2
|
||
fild word[ebp+14] ;y1
|
||
fild word[ebp+10] ;y0
|
||
fild word[ebp+16] ;x2
|
||
fild word[ebp+12] ;x1
|
||
fild word[ebp+8] ;x0
|
||
fld dword[delt_t]
|
||
fldz ;t=.0
|
||
|
||
@@:
|
||
fld1
|
||
fsub st0,st1 ;1.-t
|
||
fmul st0,st0 ;pow(1.-t,2)
|
||
fmul st0,st3 ;...*x0
|
||
fstp dword[opr_param]
|
||
|
||
fld1
|
||
fsub st0,st1 ;1.-t
|
||
fmul st0,st1 ;(1.-t)*t
|
||
fadd st0,st0
|
||
fmul st0,st4 ;...*x1
|
||
mov edi,dword[opr_param]
|
||
fstp dword[opr_param]
|
||
|
||
fldz
|
||
fadd st0,st1 ;0+t
|
||
fmul st0,st0 ;t*t
|
||
fmul st0,st5 ;...*x2
|
||
|
||
fadd dword[opr_param]
|
||
mov dword[opr_param],edi
|
||
fadd dword[opr_param]
|
||
fistp word[v_poi_0] ;x
|
||
|
||
fld1
|
||
fsub st0,st1 ;1.-t
|
||
fmul st0,st0 ;pow(1.-t,2)
|
||
fmul st0,st6 ;...*y0
|
||
fstp dword[opr_param]
|
||
|
||
fld1
|
||
fsub st0,st1 ;1.-t
|
||
fmul st0,st1 ;(1.-t)*t
|
||
fadd st0,st0
|
||
fmul st0,st7 ;...*y1
|
||
mov edi,dword[opr_param]
|
||
fstp dword[opr_param]
|
||
|
||
fldz
|
||
fadd st0,st1 ;0+t
|
||
fmul st0,st0 ;t*t
|
||
fimul word[ebp+18] ;...*y2
|
||
|
||
fadd dword[opr_param]
|
||
mov dword[opr_param],edi
|
||
fadd dword[opr_param]
|
||
fistp word[v_poi_0+2] ;y
|
||
|
||
mov eax,1
|
||
mov bx,word[v_poi_0+2]
|
||
mov cx,word[v_poi_0]
|
||
call dword[fun_draw_pixel]
|
||
|
||
fadd st0,st1 ;t+dt
|
||
|
||
fld1
|
||
fcomp
|
||
fstsw ax
|
||
sahf
|
||
jae @b
|
||
|
||
.end_draw:
|
||
; btr word[opt_bez],0 ;снимаем флаг рисования прямой линии с кривой Безье
|
||
; btr word[opt_bez],1
|
||
;and word[opt_bez],0xfffc
|
||
popad
|
||
|
||
mov esp,ebp
|
||
pop ebp
|
||
ret 16
|
||
|
||
delt_t dd 0.05 ;шаг для параметра t из кривой Безье
|
||
|
||
;функция рисующая сегмент кривуй Безье по 3-м точкам, при этом координаты
|
||
; 1-й и 3-й точки смещаются ко 2-й точке, позволяя рисовать длинную кривую из нескольких кусков
|
||
;input:
|
||
; [esp+8] = p0
|
||
; [esp+12] = p1
|
||
; [esp+16] = p2
|
||
; [esp+20] = color
|
||
align 4
|
||
cruve_bezier_del2:
|
||
; btr word[opt_bez],1 ;test
|
||
; ret 16 ;test
|
||
push ebp
|
||
mov ebp,esp
|
||
|
||
pushad
|
||
;jmp cruve_bezier.end_draw
|
||
|
||
bt word[opt_bez],1 ;проверяем флаг рисования отрезка для 3-й точки (p2)
|
||
jae @f
|
||
push dword[ebp+20] ;line color
|
||
push dword[ebp+8]
|
||
@@:
|
||
|
||
;********* высчитывание усредненных координат *********
|
||
mov ax,word[ebp+8] ;x0
|
||
add ax,word[ebp+12]
|
||
shr ax,1
|
||
bt ax,14
|
||
jae @f
|
||
or ax,0x8000
|
||
@@:
|
||
mov word[ebp+8],ax
|
||
|
||
mov ax,word[ebp+10] ;y0
|
||
add ax,word[ebp+14]
|
||
shr ax,1
|
||
bt ax,14
|
||
jae @f
|
||
or ax,0x8000
|
||
@@:
|
||
mov word[ebp+10],ax
|
||
|
||
btr word[opt_bez],1 ;проверяем флаг рисования отрезка для 3-й точки (p2)
|
||
jae @f
|
||
push dword[ebp+8]
|
||
call line_brs ;рисуем прямой отрезок
|
||
@@:
|
||
bt word[opt_bez],0 ;проверяем флаг рисования отрезка для 1-й точки (p0)
|
||
jae @f
|
||
push dword[ebp+20] ;line color
|
||
push dword[ebp+16]
|
||
@@:
|
||
|
||
mov ax,word[ebp+16] ;x2
|
||
add ax,word[ebp+12]
|
||
shr ax,1
|
||
bt ax,14
|
||
jae @f
|
||
or ax,0x8000
|
||
@@:
|
||
mov word[ebp+16],ax
|
||
|
||
mov ax,word[ebp+18] ;y2
|
||
add ax,word[ebp+14]
|
||
shr ax,1
|
||
bt ax,14
|
||
jae @f
|
||
or ax,0x8000
|
||
@@:
|
||
mov word[ebp+18],ax
|
||
|
||
btr word[opt_bez],0 ;проверяем флаг рисования отрезка для 1-й точки (p0)
|
||
jae @f
|
||
push dword[ebp+16]
|
||
call line_brs ;рисуем прямой отрезок
|
||
@@:
|
||
|
||
;jmp cruve_bezier.end_draw
|
||
;********* переход на основную функцию *********
|
||
jmp cruve_bezier.beg_fun
|
||
|
||
|
||
;функция рисующая текст
|
||
;input:
|
||
; dword[ebp+8] = pointer to vector font data
|
||
; dword[ebp+12] = pointer to text param struct (color, x,y, angle, scale, ...)
|
||
; dword[ebp+16] = text string (0 - end string)
|
||
align 4
|
||
draw_text:
|
||
push ebp
|
||
mov ebp,esp
|
||
pushad
|
||
mov eax,dword[ebp+8]
|
||
mov ebx,dword[ebp+12]
|
||
mov edx,dword[ebp+16]
|
||
|
||
mov esi,ebx
|
||
add esi,4 ;skeep color
|
||
mov edi,text_point
|
||
mov ecx,12 ;(x+y+a)*3
|
||
rep movsb ;copy base point
|
||
|
||
finit
|
||
fild word[ebx+12+4]
|
||
fdiv dword[eax+4] ;sumbol height
|
||
fstp dword[text_point.s]
|
||
|
||
mov edi,dword[ebx];get color
|
||
xor esi,esi ;line number
|
||
;------------------------
|
||
@@:
|
||
cmp byte[edx],0
|
||
je @f
|
||
xor ecx,ecx
|
||
mov cl,byte[edx]
|
||
shl cx,2 ;cx*=4
|
||
add cx,32
|
||
add ecx,eax
|
||
|
||
;mov ecx,eax
|
||
;add ecx,32+256*4+4
|
||
|
||
push edi ;color
|
||
push dword[ecx] ;copy sumbol pointer
|
||
pop ecx
|
||
add ecx,eax ;добавляем к ссылке на символ смещение начала самого шрифта
|
||
push dword ecx
|
||
push dword text_point ;output point
|
||
call draw_poly_line
|
||
|
||
cmp byte[edx],13
|
||
je .new_line
|
||
finit
|
||
fld dword[eax] ;sumbol width
|
||
fmul dword[text_point.s]
|
||
|
||
fld dword[text_point.a]
|
||
fcos
|
||
fmul st0,st1
|
||
fadd dword[text_point.x]
|
||
fstp dword[text_point.x]
|
||
|
||
fld dword[text_point.a]
|
||
fsin
|
||
fmul st0,st1
|
||
fadd dword[text_point.y]
|
||
fstp dword[text_point.y]
|
||
inc edx ;move next sumbol
|
||
jmp @b
|
||
.new_line:
|
||
push edi esi
|
||
mov esi,dword[ebp+12]
|
||
add esi,4 ;skeep color
|
||
mov edi,text_point
|
||
mov ecx,8 ;(x+y)*4
|
||
rep movsb ;restore base point
|
||
pop esi edi
|
||
|
||
inc esi
|
||
mov dword[opr_param],esi
|
||
|
||
finit
|
||
fld dword[eax+4] ;sumbol height
|
||
fmul dword[text_point.s]
|
||
fimul dword[opr_param]
|
||
|
||
fld1
|
||
fld1
|
||
fadd st0,st1
|
||
fldpi
|
||
fdiv st0,st1
|
||
fadd dword[text_point.a]
|
||
fcos
|
||
; fld dword[eax+4] ;sumbol height
|
||
; fmul dword[text_point.s]
|
||
fmul st0,st3
|
||
fadd dword[text_point.x]
|
||
fstp dword[text_point.x]
|
||
|
||
fld1
|
||
fld1
|
||
fadd st0,st1
|
||
fldpi
|
||
fdiv st0,st1
|
||
fadd dword[text_point.a]
|
||
fsin
|
||
; fld dword[eax+4] ;sumbol height
|
||
; fmul dword[text_point.s]
|
||
fmul st0,st5
|
||
fadd dword[text_point.y]
|
||
fstp dword[text_point.y]
|
||
inc edx ;move next sumbol
|
||
jmp @b
|
||
@@:
|
||
popad
|
||
pop ebp
|
||
ret 12
|
||
|
||
text_point: ;точка для вывода текста
|
||
.x dd 0.0
|
||
.y dd 0.0
|
||
.a dd 0.0 ;angle
|
||
.s dd 1.0 ;scale
|
||
|
||
|
||
;функция для конвертирования координат из декартовой системы координат в полярную
|
||
;input:
|
||
; dword[ebp+8] = pointer to contur
|
||
align 4
|
||
convert_contur:
|
||
push ebp
|
||
mov ebp,esp
|
||
|
||
push eax ebx ecx
|
||
mov ebx,dword[ebp+8]
|
||
finit
|
||
.new_contur:
|
||
mov cx,word[ebx]
|
||
or word[ebx],VECT_POINTS_IS_POLAR
|
||
|
||
add ebx,2
|
||
btr cx,15 ;VECT_PARAM_COLOR
|
||
jae @f
|
||
add ebx,4
|
||
@@:
|
||
btr cx,14 ;VECT_PARAM_PROP_L
|
||
jae @f
|
||
add ebx,4
|
||
@@:
|
||
btr cx,12 ;проверка конвертированных
|
||
jae @f
|
||
and ecx,0xfff
|
||
cmp cx,0
|
||
je .end_contur
|
||
shl ecx,3
|
||
add ebx,ecx
|
||
jmp .new_contur
|
||
@@:
|
||
and cx,0xfff
|
||
cmp cx,0
|
||
je .end_contur
|
||
mov eax,opr_param
|
||
|
||
@@:
|
||
cmp cx,0
|
||
je .new_contur
|
||
dec cx
|
||
fld dword[ebx]
|
||
fistp word[eax]
|
||
fld dword[ebx+4]
|
||
fistp word[eax+2]
|
||
|
||
push dword[eax]
|
||
call opred2i
|
||
push dword[o_ang]
|
||
pop dword[ebx] ;x(n)
|
||
push dword[o_len]
|
||
pop dword[ebx+4] ;y(n)
|
||
|
||
add ebx,8 ;move next coord
|
||
jmp @b
|
||
.end_contur:
|
||
pop ecx ebx eax ebp
|
||
ret 4
|
||
|
||
;функция для рисования контуров, состоящих из разных наборов точек (прямые, Безье)
|
||
;input:
|
||
; [esp+8] = x0,y0,a0,s0 - параметры центральной точки: координаты, угол поворота, масштаб
|
||
; [esp+12] = contur - контур, заданный координатами точек
|
||
; [esp+16] = color - первоначальный цвет
|
||
align 4
|
||
draw_poly_line:
|
||
push ebp
|
||
mov ebp,esp
|
||
pushad
|
||
mov eax,dword[ebp+8]
|
||
mov ebx,dword[ebp+12]
|
||
mov edx,dword[ebp+16]
|
||
|
||
bt word[ebx],12 ;VECT_POINTS_IS_POLAR
|
||
jc @f
|
||
push dword ebx
|
||
call convert_contur
|
||
@@:
|
||
|
||
mov word[opt_bez],0 ;clear draw options
|
||
|
||
finit
|
||
fld dword[eax] ;x0 (st4)
|
||
fld dword[eax+4];y0 (st3)
|
||
fld dword[eax+8];a0 (st2)
|
||
fld dword[eax+12];s0(st1)
|
||
|
||
mov si,0 ;сплошная линия, без прерываний
|
||
|
||
.new_contur:
|
||
|
||
mov cx,word[ebx]
|
||
mov edi,ebx ;резервное сохранение начала контура
|
||
|
||
add ebx,2
|
||
btr cx,15 ;VECT_PARAM_COLOR
|
||
jae @f
|
||
mov edx,dword[ebx]
|
||
add ebx,4
|
||
@@:
|
||
btr cx,14 ;VECT_PARAM_PROP_L
|
||
jae @f
|
||
mov esi,dword[ebx] ;получаем новые параметры линии
|
||
add ebx,4
|
||
btr si,9 ;VECT_CONT_BEZIER = 0x200
|
||
jc .BezB ;дальше контур Безье, не линейный
|
||
and si,0xff
|
||
@@:
|
||
|
||
jmp .BezE
|
||
.BezB: ;пропуск всех точек Безье
|
||
bt word[opt_bez],2
|
||
jc .after_bez_draw
|
||
or word[opt_bez],4
|
||
|
||
push edx
|
||
push edi ;начало контура - ebx
|
||
push dword[ebp+8] ;eax
|
||
call draw_poly_bezier
|
||
.after_bez_draw:
|
||
and ecx,0xfff
|
||
cmp ecx,0
|
||
je .end_contur
|
||
|
||
shl ecx,3
|
||
add ebx,ecx
|
||
jmp .new_contur
|
||
.BezE:
|
||
|
||
and cx,0xfff
|
||
cmp cx,0
|
||
je .end_contur
|
||
|
||
mov di,cx
|
||
sub di,si
|
||
|
||
fld dword[ebx] ;st0=a(n)
|
||
fadd st0,st2
|
||
fcos
|
||
fmul dword[ebx+4] ;l(n)
|
||
fmul st0,st1 ;*=scale
|
||
fadd st0,st4
|
||
fistp word[v_poi_1+2] ;x(n)
|
||
|
||
fld dword[ebx] ;st0=a(n)
|
||
fadd st0,st2
|
||
fsin
|
||
fmul dword[ebx+4] ;l(n)
|
||
fmul st0,st1 ;*=scale
|
||
fadd st0,st3
|
||
fistp word[v_poi_1] ;y(n)
|
||
dec cx
|
||
add ebx,8 ;move next coord
|
||
|
||
@@: ;---------------------------------------------------------------
|
||
push dword[v_poi_1]
|
||
pop dword[v_poi_0]
|
||
; call draw_vect_point
|
||
|
||
cmp cx,0
|
||
je .new_contur ;во избежание зацикливания
|
||
|
||
fld dword[ebx] ;st0=a(n)
|
||
fadd st0,st2
|
||
fcos
|
||
fmul dword[ebx+4] ;l(n)
|
||
fmul st0,st1 ;*=scale
|
||
fadd st0,st4
|
||
fistp word[v_poi_1+2] ;x(n)
|
||
|
||
fld dword[ebx] ;st0=a(n)
|
||
fadd st0,st2
|
||
fsin
|
||
fmul dword[ebx+4] ;l(n)
|
||
fmul st0,st1 ;*=scale
|
||
fadd st0,st3
|
||
fistp word[v_poi_1] ;y(n)
|
||
add ebx,8 ;move next coord
|
||
|
||
cmp cx,di
|
||
je .end_draw_line
|
||
|
||
push dword edx ;line color
|
||
push dword[v_poi_0]
|
||
push dword[v_poi_1]
|
||
call line_brs
|
||
|
||
loop @b
|
||
jmp .new_contur
|
||
|
||
.end_draw_line: ;-------------------------------------------------------------
|
||
sub di,si
|
||
loop @b
|
||
jmp .new_contur
|
||
|
||
.end_contur:
|
||
popad
|
||
pop ebp
|
||
ret 12
|
||
|
||
;функция рисующая полигоны состоящие из кривых Безье
|
||
;input:
|
||
; [esp+8] = x0,y0,a0,...
|
||
; [esp+12] = contur
|
||
; [esp+16] = color
|
||
align 4
|
||
draw_poly_bezier:
|
||
push ebp
|
||
mov ebp,esp
|
||
pushad
|
||
mov eax,dword[ebp+8]
|
||
mov ebx,dword[ebp+12]
|
||
mov edx,dword[ebp+16]
|
||
finit
|
||
fld dword[eax] ;x0 (st4)
|
||
fld dword[eax+4];y0 (st3)
|
||
fld dword[eax+8];a0 (st2)
|
||
fld dword[eax+12];s0(st1)
|
||
|
||
mov si,0 ;сплошная линия, без прерываний
|
||
|
||
.new_contur:
|
||
|
||
mov cx,word[ebx]
|
||
add ebx,2
|
||
btr cx,15
|
||
jae @f
|
||
mov edx,dword[ebx]
|
||
add ebx,4
|
||
@@:
|
||
btr cx,14 ;VECT_PARAM_PROP_L
|
||
jae @f
|
||
mov esi,dword[ebx] ;получаем новые параметры линии
|
||
btr si,8 ;VECT_CONT_LINE = 0x100
|
||
;jc .end_contur ;дальше контур линейный, не Безье
|
||
jae .skip
|
||
and ecx,0xfff
|
||
cmp ecx,0
|
||
je .end_contur
|
||
add ebx,4
|
||
shl ecx,3
|
||
add ebx,ecx
|
||
jmp .new_contur
|
||
.skip:
|
||
and si,0xff
|
||
add ebx,4
|
||
@@:
|
||
and ecx,0xfff
|
||
cmp ecx,0
|
||
je .end_contur
|
||
|
||
cmp si,1 ;проверка контура на 3 точки
|
||
je @f
|
||
cmp si,2
|
||
je @f
|
||
jmp .3pt
|
||
shl ecx,3
|
||
add ebx,ecx
|
||
|
||
jmp .new_contur
|
||
.3pt: ;тут контуры минимум с 3-мя точками
|
||
|
||
mov di,si
|
||
|
||
fld dword[ebx] ;st0=a(n)
|
||
fadd st0,st2
|
||
fcos
|
||
fmul dword[ebx+4] ;l(n)
|
||
fmul st0,st1 ;*=scale
|
||
fadd st0,st4
|
||
fistp word[v_poi_1+2] ;x(n)
|
||
|
||
fld dword[ebx] ;st0=a(n)
|
||
fadd st0,st2
|
||
fsin
|
||
fmul dword[ebx+4] ;l(n)
|
||
fmul st0,st1 ;*=scale
|
||
fadd st0,st3
|
||
fistp word[v_poi_1] ;y(n)
|
||
dec cx
|
||
add ebx,8 ;move next coord
|
||
cmp cx,0
|
||
je .new_contur ;во избежание зацикливания
|
||
|
||
fld dword[ebx] ;st0=a(n)
|
||
fadd st0,st2
|
||
fcos
|
||
fmul dword[ebx+4] ;l(n)
|
||
fmul st0,st1 ;*=scale
|
||
fadd st0,st4
|
||
fistp word[v_poi_2+2] ;x(n)
|
||
|
||
fld dword[ebx] ;st0=a(n)
|
||
fadd st0,st2
|
||
fsin
|
||
fmul dword[ebx+4] ;l(n)
|
||
fmul st0,st1 ;*=scale
|
||
fadd st0,st3
|
||
fistp word[v_poi_2] ;y(n)
|
||
dec cx
|
||
add ebx,8 ;move next coord
|
||
|
||
or word[opt_bez],1 ;begin line
|
||
.bez_cycl: ;---------------------------------------------------------
|
||
|
||
dec di
|
||
push dword[v_poi_1]
|
||
pop dword[v_poi_0]
|
||
push dword[v_poi_2]
|
||
pop dword[v_poi_1]
|
||
|
||
cmp cx,0
|
||
je .new_contur ;во избежание зацикливания
|
||
|
||
fld dword[ebx] ;st0=a(n)
|
||
fadd st0,st2
|
||
fcos
|
||
fmul dword[ebx+4] ;l(n)
|
||
fmul st0,st1 ;*=scale
|
||
fadd st0,st4
|
||
fistp word[v_poi_2+2] ;x(n)
|
||
|
||
fld dword[ebx] ;st0=a(n)
|
||
fadd st0,st2
|
||
fsin
|
||
fmul dword[ebx+4] ;l(n)
|
||
fmul st0,st1 ;*=scale
|
||
fadd st0,st3
|
||
fistp word[v_poi_2] ;y(n)
|
||
add ebx,8 ;move next coord
|
||
|
||
cmp di,2
|
||
jne @f
|
||
or word[opt_bez],2 ;end line
|
||
@@:
|
||
cmp cx,1
|
||
jne @f
|
||
or word[opt_bez],2 ;end line
|
||
@@:
|
||
dec si
|
||
cmp di,si
|
||
jne @f
|
||
or word[opt_bez],1 ;begin line
|
||
@@:
|
||
inc si
|
||
cmp di,si
|
||
je @f
|
||
cmp di,1
|
||
je .end_draw_line
|
||
push dword edx ;line color
|
||
push dword[v_poi_0]
|
||
push dword[v_poi_1]
|
||
push dword[v_poi_2]
|
||
call cruve_bezier_del2
|
||
finit
|
||
fld dword[eax] ;x0 (st4)
|
||
fld dword[eax+4];y0 (st3)
|
||
fld dword[eax+8];a0 (st2)
|
||
fld dword[eax+12];s0(st1)
|
||
|
||
jmp @f
|
||
.end_draw_line: ;-------------------------------------------------------------
|
||
mov di,si
|
||
inc di
|
||
@@:
|
||
|
||
;loop .bez_cycl ;@b
|
||
dec cx
|
||
cmp cx,0
|
||
jg .bez_cycl
|
||
|
||
jmp .new_contur
|
||
.end_contur:
|
||
popad
|
||
pop ebp
|
||
ret 12
|
||
|
||
|
||
;функция принимает координаты точки x,y и определяет угол и длину
|
||
;input:
|
||
; ebp+8 = adress int coord x
|
||
; ebp+10 = adress int coord y
|
||
align 4
|
||
opred2i:
|
||
push ebp
|
||
mov ebp,esp
|
||
finit
|
||
fild word [ebp+8]
|
||
fmul st0,st0 ;st0=x^2
|
||
fild word [ebp+10]
|
||
fmul st0,st0 ;st0=y^2
|
||
fadd st0,st1
|
||
fsqrt
|
||
fst dword [o_len]
|
||
cmp dword [o_len],0
|
||
jne @f
|
||
mov dword [o_ang],0
|
||
jmp .retf
|
||
@@:
|
||
fild word [ebp+8]
|
||
fdiv dword [o_len]
|
||
call acos
|
||
|
||
cmp word [ebp+10],0
|
||
jl @f
|
||
fst [o_ang] ;a=acos(x/l);
|
||
jmp .retf
|
||
@@:
|
||
|
||
fldpi
|
||
fadd st0,st0 ;st0=2*pi
|
||
fsub st0,st1 ;st0=2*pi-aac
|
||
fst [o_ang] ;a=st0;
|
||
.retf:
|
||
pop ebp
|
||
ret 4
|
||
|
||
;функция определяющая расстояние между точками, результат попадает в o_len
|
||
;input:
|
||
; ebp+8 = p0
|
||
; ebp+12 = p1
|
||
align 4
|
||
line_len4i:
|
||
push ebp
|
||
mov ebp,esp
|
||
|
||
finit
|
||
fild word [ebp+8]
|
||
fisub word [ebp+12]
|
||
fmul st0,st0 ;st0=x^2
|
||
fild word [ebp+10]
|
||
fisub word [ebp+14]
|
||
fmul st0,st0 ;st0=y^2
|
||
fadd st0,st1
|
||
fsqrt
|
||
fstp dword [o_len]
|
||
|
||
pop ebp
|
||
ret 8
|
||
|
||
;функция для нахождения арккосинуса
|
||
;input:
|
||
; st0 = float value
|
||
align 4
|
||
acos:
|
||
fld1
|
||
fadd st, st1
|
||
fld1
|
||
fsub st, st2
|
||
fmulp st1, st
|
||
fsqrt
|
||
fxch st1
|
||
fpatan
|
||
ret
|
||
|
||
o_len dd ? ;длина
|
||
o_ang dd ? ;угол порота в радианах
|
||
opr_param dd ?
|
||
v_poi_0 dd ?
|
||
v_poi_1 dd ?
|
||
v_poi_2 dd ?
|
||
opt_bez dw ? ;опции рисования кусков для кривой Безье
|
||
|
||
align 16
|
||
EXPORTS:
|
||
dd sz_buf_create, buf_create
|
||
dd sz_set_active_buf, set_active_buf
|
||
dd sz_buf_clear, buf_clear
|
||
dd sz_draw_buf, draw_buf
|
||
dd sz_buf_delete, buf_delete
|
||
|
||
dd sz_line, line_brs
|
||
dd sz_cruve_bezier, cruve_bezier
|
||
dd sz_conv, convert_contur
|
||
dd sz_draw, draw_poly_line
|
||
dd sz_opred2i, opred2i
|
||
dd sz_line_len4i, line_len4i
|
||
dd sz_draw_text, draw_text
|
||
dd sz_o_len, o_len
|
||
dd sz_o_ang, o_ang
|
||
dd 0,0
|
||
sz_buf_create db 'vect_buf_create',0
|
||
sz_set_active_buf db 'vect_buf_set_active',0
|
||
sz_buf_clear db 'vect_buf_clear',0
|
||
sz_draw_buf db 'vect_buf_draw',0
|
||
sz_buf_delete db 'vect_buf_delete',0
|
||
|
||
sz_line db 'vect_line',0
|
||
sz_cruve_bezier db 'vect_c_bezier',0
|
||
sz_conv db 'vect_conv_cont',0
|
||
sz_draw db 'vect_draw_cont',0
|
||
sz_opred2i db 'vect_opred2i',0
|
||
sz_line_len4i db 'vect_line_len4i',0
|
||
sz_draw_text db 'vect_draw_text',0
|
||
sz_o_len db 'vect_o_len',0
|
||
sz_o_ang db 'vect_o_ang',0
|
||
|