1 /* Project 16 Source Code~
\r
2 * Copyright (C) 2012-2016 sparky4 & pngwen & andrius4669 & joncampbell123 & yakui-lover
\r
4 * This file is part of Project 16.
\r
6 * Project 16 is free software; you can redistribute it and/or modify
\r
7 * it under the terms of the GNU General Public License as published by
\r
8 * the Free Software Foundation; either version 3 of the License, or
\r
9 * (at your option) any later version.
\r
11 * Project 16 is distributed in the hope that it will be useful,
\r
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
\r
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
\r
14 * GNU General Public License for more details.
\r
16 * You should have received a copy of the GNU General Public License
\r
17 * along with this program. If not, see <http://www.gnu.org/licenses/>, or
\r
18 * write to the Free Software Foundation, Inc., 51 Franklin Street,
\r
19 * Fifth Floor, Boston, MA 02110-1301 USA.
\r
23 #include "src/lib/16_sprit.h"
\r
25 char* get_curr_anim_name(struct sprite *spri)
\r
27 // Retrive animation name list
\r
28 struct vrs_header far *vrs = spri->spritesheet->vrs_hdr;
\r
29 uint32_t far *anim_names_offsets = (uint32_t far *)
\r
31 vrs->offset_table[VRS_HEADER_OFFSET_ANIMATION_NAME_LIST]);
\r
33 return (char *)(vrs + anim_names_offsets[spri->curr_anim]);
\r
36 void init_anim(struct sprite *spri, int anim_index)
\r
38 struct vrs_header far *vrs = spri->spritesheet->vrs_hdr;
\r
39 uint32_t far *anim_lists_offsets = (uint32_t far *)
\r
41 vrs->offset_table[VRS_HEADER_OFFSET_ANIMATION_LIST]);
\r
42 struct vrs_animation_list_entry_t far *anim_list = (struct vrs_animation_list_entry_t far *)
\r
44 anim_lists_offsets[anim_index]);
\r
46 // Upon new animation, start from the first sprite in it
\r
47 spri->curr_anim = anim_index;
\r
48 spri->curr_anim_spri = 0;
\r
49 spri->curr_spri_id = anim_list[0].sprite_id;
\r
50 spri->delay = anim_list[0].delay;
\r
52 spri->curr_anim_list = anim_list;
\r
55 int set_anim_by_id(struct sprite *spri, int anim_id)
\r
57 int new_anim_index = 0;
\r
59 struct vrs_header far *vrs = spri->spritesheet->vrs_hdr;
\r
60 // Retruve animation ids list
\r
61 uint16_t far *anim_ids = (uint16_t far *)
\r
63 vrs->offset_table[VRS_HEADER_OFFSET_ANIMATION_ID_LIST]);
\r
65 // Loop through animation id untill match or end of list
\r
66 while(iter_id = anim_ids[new_anim_index])
\r
68 // Return on successful match
\r
69 if (iter_id == anim_id)
\r
71 init_anim(spri, new_anim_index);
\r
79 void print_anim_ids(struct sprite *spri)
\r
81 int new_anim_index = 0;
\r
83 struct vrs_header far *vrs = spri->spritesheet->vrs_hdr;
\r
84 // Retruve animation ids list
\r
85 uint16_t far *anim_ids = (uint16_t far *)
\r
87 vrs->offset_table[VRS_HEADER_OFFSET_ANIMATION_ID_LIST]);
\r
89 if(!anim_ids[new_anim_index])
\r
91 // Loop through animation id untill match or end of list
\r
92 while(iter_id = anim_ids[new_anim_index])
\r
94 printf("s%d ", iter_id);
\r
99 void animate_spri(entity_t *enti, video_t *video)
\r
101 #define INC_PER_FRAME if(enti->q&1) enti->persist_aniframe++; if(enti->persist_aniframe>4) enti->persist_aniframe = 1;
\r
103 #define GVARVIDEO video
\r
104 #define VMEMPAGESIZE2 GVARVIDEO->page[0].pagesize+GVARVIDEO->page[1].pagesize
\r
105 #define VMEMPAGEDATA2 GVARVIDEO->page[2].data
\r
106 unsigned int i,o,o2; int j;
\r
109 VGA_RAM_PTR omemptr;
\r
111 // Depending on delay, update indices
\r
112 //#define FRAME1 modexDrawSpriteRegion(pip[(pip->video->p)].page, x, y, 48, player[pn].enti.dire, 24, 32, PLAYERBMPDATAPTR);
\r
113 //#define FRAME2 modexDrawSpriteRegion(pip[(pip->video->p)].page, x, y, 24, player[pn].enti.dire, 24, 32, PLAYERBMPDATAPTR); stand
\r
114 //#define FRAME3 modexDrawSpriteRegion(pip[(pip->video->p)].page, x, y, 0, player[pn].enti.dire, 24, 32, PLAYERBMPDATAPTR);
\r
115 //#define FRAME4 modexDrawSpriteRegion(pip[(pip->video->p)].page, x, y, 24, player[pn].enti.dire, 24, 32, PLAYERBMPDATAPTR); stand
\r
116 switch(enti->spri->delay)
\r
118 // Delay = 0 means that sprite should loop. Nothing to change here
\r
122 // Delay = 1 means that on next time unit sprite should be changed
\r
125 enti->spri->curr_anim_spri++;
\r
127 // If we hit the end of an animation sequence, restart it
\r
128 if(!( enti->spri->curr_spri_id = enti->spri->curr_anim_list[enti->spri->curr_anim_spri].sprite_id)){
\r
129 enti->spri->curr_anim_spri = 0;
\r
130 enti->spri->curr_spri_id = enti->spri->curr_anim_list[enti->spri->curr_anim_spri].sprite_id;
\r
132 enti->spri->delay = enti->spri->curr_anim_list[enti->spri->curr_anim_spri].delay;
\r
134 // Delay > 1 means that we should not change sprite yet. Decrease delay
\r
136 enti->spri->delay--;
\r
143 omemptr = (VGA_RAM_PTR)video->page[0].data;// save original mem ptr
\r
148 j = get_vrl_by_id(enti->spri->spritesheet, enti->spri->curr_spri_id, enti->spri->sprite_vrl_cont);
\r
151 //Quit (gv, "Error retriving required sprite");
\r
155 // render box bounds. y does not need modification, but x and width must be multiple of 4
\r
156 if(!GVARVIDEO->rss){
\r
157 if (x >= enti->overdraww) rx = (x - enti->overdraww) & (~3);
\r
158 else rx = -(GVARVIDEO->page[0].dx);
\r
159 if (y >= enti->overdrawh) ry = (y - enti->overdrawh);
\r
160 else ry = -(GVARVIDEO->page[0].dy);
\r
161 h = enti->spri->sprite_vrl_cont->vrl_header->height + enti->overdrawh + y - ry;
\r
162 w = (x + enti->spri->sprite_vrl_cont->vrl_header->width + (enti->overdraww*2) + 3 - rx) & (~3) - enti->overdraww;//round up
\r
163 if ((rx+w) > GVARVIDEO->page[0].width) w = GVARVIDEO->page[0].width-rx;
\r
164 if ((ry+h) > GVARVIDEO->page[0].height) h = GVARVIDEO->page[0].height-ry;
\r
166 // block copy pattern to where we will draw the sprite
\r
167 vga_setup_wm1_block_copy();
\r
168 o2 = VMEMPAGESIZE2;
\r
169 o = (0x10000UL - (uint16_t)VMEMPAGEDATA2) + (ry * GVARVIDEO->page[0].stridew) + (rx >> 2); // source offscreen
\r
170 for (i=0;i < h;i++,o += GVARVIDEO->page[0].stridew,o2 += (w >> 2)) vga_wm1_mem_block_copy(o2,o,w >> 2);
\r
172 // must restore Write Mode 0/Read Mode 0 for this code to continue drawing normally
\r
173 vga_restore_rm0wm0();
\r
175 // replace VGA stride with our own and mem ptr. then sprite rendering at this stage is just (0,0)
\r
176 vga_state.vga_draw_stride = w >> 2;
\r
177 vga_state.vga_graphics_ram = omemptr + VMEMPAGESIZE2;
\r
178 }else{ rx=ry=w=h=0; vga_state.vga_graphics_ram = (VGA_RAM_PTR)video->page[0].data; }
\r
179 vga_state.vga_draw_stride_limit = (GVARVIDEO->page[0].width + 3 - x) >> 2;//round up
\r
181 // then the sprite. note modding ram ptr means we just draw to (x&3,0)
\r
183 modexClearRegion(&GVARVIDEO->page[0], x, y, 16, 32, 1);
\r
185 draw_vrl1_vgax_modex(
\r
188 enti->spri->sprite_vrl_cont->vrl_header,
\r
189 enti->spri->sprite_vrl_cont->line_offsets,
\r
190 enti->spri->sprite_vrl_cont->buffer + sizeof(struct vrl1_vgax_header),
\r
191 enti->spri->sprite_vrl_cont->data_size
\r
194 if(!GVARVIDEO->rss){
\r
196 vga_state.vga_graphics_ram = omemptr;
\r
198 // block copy to visible RAM from offscreen
\r
199 vga_setup_wm1_block_copy();
\r
200 o = VMEMPAGESIZE2; // source offscreen
\r
201 o2 = (ry * GVARVIDEO->page[0].stridew) + (rx >> 2); // dest visible (original stride)
\r
202 for (i=0;i < h;i++,o += vga_state.vga_draw_stride,o2 += GVARVIDEO->page[0].stridew) vga_wm1_mem_block_copy(o2,o,w >> 2);
\r
203 // must restore Write Mode 0/Read Mode 0 for this code to continue drawing normally
\r
204 vga_restore_rm0wm0();
\r
207 vga_state.vga_draw_stride_limit = vga_state.vga_draw_stride = GVARVIDEO->page[0].stridew;
\r
209 vga_state.vga_graphics_ram = video->omemptr;
\r