#help_index "Graphics/Sprite;Sprites" /* CSprites are stored as a sequence of var length operations with a 1-byte type leading each operation. They are stored, one after another, in a chunk of memory terminated by a zero. Sprite3() shows how the CSprite unions are used. SpriteElemSize() will return the size of a single element, while SpriteSize() will return the size of an entire list. Look at sprite_elem_base_sizes. See ::/Apps/GrModels for an example of making CSprite by hand. It uses SPT_MESH, one of the most complicated. */ public U0 Sprite3(CDC *dc=gr.dc, I64 x, I64 y, I64 z, U8 *elems, Bool just_one_elem=FALSE) {//Plot a sprite into a CDC. CSprite *tmpg = elems - offset(CSprite.start); I64 i, j, k, x1, y1, z1, x2, y2, *old_r, *r2, old_flags = dc->flags, old_pen_width = dc->thick; I32 *ptr; CColorROPU32 old_color = dc->color; CDC *img; CD3I32 *p, *p2; CGrSym old_sym; MemCopy(&old_sym, &dc->sym, sizeof(CGrSym)); if (dc->flags & DCF_LOCATE_NEAREST) dc->nearest_dist = I64_MAX; while (tmpg->type & SPG_TYPE_MASK) { switch (tmpg->type & SPG_TYPE_MASK) { case SPT_COLOR: dc->color = dc->color & ~(COLORROP_COLORS_MASK | ROPF_DITHER) | tmpg->c.color; break; case SPT_DITHER_COLOR: dc->color = dc->color & ~COLORROP_COLORS_MASK | tmpg->d.dither_color.u8[0] | tmpg->d.dither_color.u8[1] << COLORROP_BITS | ROPF_DITHER; break; case SPT_THICK: dc->thick = tmpg->t.thick; DCThickScale(dc); break; case SPT_TRANSFORM_ON: if (!(dc->flags & DCF_TRANSFORMATION)) { x -= dc->x; y -= dc->y; z -= dc->z; } dc->flags |= DCF_TRANSFORMATION; break; case SPT_TRANSFORM_OFF: if (dc->flags & DCF_TRANSFORMATION) { x += dc->x; y += dc->y; z += dc->z; } dc->flags &= ~DCF_TRANSFORMATION; break; case SPT_PT: GrPlot3(dc, tmpg->p.x1 + x, tmpg->p.y1 + y, z); break; case SPT_TEXT: GrPrint3(dc, tmpg->ps.x1 + x, tmpg->ps.y1 + y, z, "%s", tmpg->ps.st); break; case SPT_TEXT_BOX: GrTextBox3(dc, tmpg->ps.x1 + x, tmpg->ps.y1 + y, z, tmpg->ps.st); break; case SPT_TEXT_DIAMOND: GrTextDiamond3(dc, tmpg->ps.x1 + x, tmpg->ps.y1 + y, z, tmpg->ps.st); break; case SPT_FLOOD_FILL: GrFloodFill3(dc, tmpg->p.x1 + x, tmpg->p.y1 + y, z, FALSE); break; case SPT_FLOOD_FILL_NOT: i = dc->color; dc->color = dc->color.c0; GrFloodFill3(dc, tmpg->p.x1 + x, tmpg->p.y1 + y, z, TRUE); dc->color = i; break; case SPT_SHIFT: x += tmpg->p.x1; y += tmpg->p.y1; break; case SPT_LINE: GrLine3(dc, tmpg->pp.x1 + x, tmpg->pp.y1 + y, z, tmpg->pp.x2 + x, tmpg->pp.y2 + y, z); break; case SPT_ARROW: GrArrow3(dc, tmpg->pp.x1 + x, tmpg->pp.y1 + y, z, tmpg->pp.x2 + x, tmpg->pp.y2 + y, z); break; case SPT_PLANAR_SYMMETRY: if (DCSymmetry3Set(dc, tmpg->pp.x1 + x, tmpg->pp.y1 + y, z, tmpg->pp.x2 + x, tmpg->pp.y2 + y, z, tmpg->pp.x2 + x, tmpg->pp.y2 + y, z + 1)) dc->flags |= DCF_SYMMETRY; else dc->flags &= ~DCF_SYMMETRY; break; case SPT_BITMAP: img = CAlloc(sizeof(CDC)); img->width = tmpg->pwhu.width; img->width_internal = (tmpg->pwhu.width + 7) & ~7; img->height = tmpg->pwhu.height; img->body = &tmpg->pwhu.u; img->dc_signature = DCS_SIGNATURE_VAL; GrBlot3(dc, tmpg->pwhu.x1 + x, tmpg->pwhu.y1 + y, z, img); Free(img); break; case SPT_RECT: GrRect3(dc, tmpg->pp.x1 + x, tmpg->pp.y1 + y, z, tmpg->pp.x2 - tmpg->pp.x1, tmpg->pp.y2 - tmpg->pp.y1); break; case SPT_ROTATED_RECT: x1 = tmpg->ppa.x1 + x; y1 = tmpg->ppa.y1 + y; z1 = z; Mat4x4MulXYZ(dc->r, &x1, &y1, &z1); old_r = dc->r; dc->flags |= DCF_TRANSFORMATION; r2 = Mat4x4IdentNew; Mat4x4RotZ(r2, -tmpg->ppa.angle); Mat4x4TranslationEqu(r2, x1, y1, z1); DCMat4x4Set(dc, Mat4x4MulMat4x4New(old_r, r2)); GrRect3(dc, 0, 0, 0, tmpg->ppa.x2 - tmpg->ppa.x1, tmpg->ppa.y2 - tmpg->ppa.y1); Free(dc->r); Free(r2); DCMat4x4Set(dc, old_r); dc->flags = dc->flags & ~DCF_TRANSFORMATION | old_flags; break; case SPT_CIRCLE: GrCircle3(dc, tmpg->pr.x1 + x, tmpg->pr.y1 + y, z, tmpg->pr.radius); break; case SPT_ELLIPSE: GrEllipse3(dc, tmpg->pwha.x1 + x, tmpg->pwha.y1 + y, z, tmpg->pwha.width, tmpg->pwha.height, tmpg->pwha.angle); break; case SPT_POLYGON: GrRegPoly3(dc, tmpg->pwhas.x1 + x, tmpg->pwhas.y1 + y, z, tmpg->pwhas.width, tmpg->pwhas.height, tmpg->pwhas.sides, tmpg->pwhas.angle); break; case SPT_POLYLINE: ptr = &tmpg->nu.u; x1 = ptr[0]; y1 = ptr[1]; for (i = 1; i < tmpg->nu.num; i++) { x2 = ptr[i << 1]; y2 = ptr[i << 1 + 1]; GrLine3(dc, x1 + x, y1 + y, z, x2 + x, y2 + y, z); x1 = x2; y1 = y2; } break; case SPT_POLYPT: x1 = tmpg->npu.x; y1 = tmpg->npu.y; ptr = &tmpg->npu.u; k = tmpg->npu.num * 3; GrPlot3(dc, x1 + x, y1 + y, z); for (i = 0; i < k; i += 3) { j = BFieldExtU32(ptr, i, 3); x1 += gr_x_offsets[j]; y1 += gr_y_offsets[j]; GrPlot3(dc, x1 + x, y1 + y, z); } break; start: p2 = p = MAlloc(tmpg->nu.num * sizeof(CD3I32)); MemCopy(p, &tmpg->nu.u, tmpg->nu.num * sizeof(CD3I32)); for (i = 0; i < tmpg->nu.num; i++, p2++) { p2->x += x; p2->y += y; p2->z += z; } case SPT_BSPLINE2: Gr2BSpline3(dc, p, tmpg->nu.num, FALSE); break; case SPT_BSPLINE3: Gr3BSpline3(dc, p, tmpg->nu.num, FALSE); break; case SPT_BSPLINE2_CLOSED: Gr2BSpline3(dc, p, tmpg->nu.num, TRUE); break; case SPT_BSPLINE3_CLOSED: Gr3BSpline3(dc, p, tmpg->nu.num, TRUE); break; end: Free(p); break; case SPT_MESH: p2 = p = MAlloc(tmpg->mu.vertex_count * sizeof(CD3I32)); MemCopy(p, &tmpg->mu.u, tmpg->mu.vertex_count * sizeof(CD3I32)); for (i = 0; i < tmpg->mu.vertex_count; i++, p2++) { p2->x += x; p2->y += y; p2->z += z; } Gr3Mesh(dc, tmpg->mu.vertex_count, p, tmpg->mu.tri_count, (&tmpg->mu.u)(U8 *) + sizeof(CD3I32) * tmpg->mu.vertex_count); Free(p); break; case SPT_SHIFTABLE_MESH: if (dc->flags & DCF_TRANSFORMATION) { dc->x += tmpg->pmu.x; dc->y += tmpg->pmu.y; dc->z += tmpg->pmu.z; x1 = x; y1 = y; z1 = z; } else { x1 = tmpg->pmu.x + x; y1 = tmpg->pmu.y + y; z1 = tmpg->pmu.z + z; } p2 = p = MAlloc(tmpg->pmu.vertex_count * sizeof(CD3I32)); MemCopy(p, &tmpg->pmu.u, tmpg->pmu.vertex_count * sizeof(CD3I32)); for (i = 0; i < tmpg->pmu.vertex_count; i++, p2++) { p2->x += x1; p2->y += y1; p2->z += z1; } Gr3Mesh(dc, tmpg->pmu.vertex_count, p, tmpg->pmu.tri_count, (&tmpg->pmu.u)(U8 *) + sizeof(CD3I32) * tmpg->pmu.vertex_count); Free(p); if (dc->flags & DCF_TRANSFORMATION) { dc->x -= tmpg->pmu.x; dc->y -= tmpg->pmu.y; dc->z -= tmpg->pmu.z; } break; } if (just_one_elem) break; tmpg(U8 *) += SpriteElemSize(tmpg); } MemCopy(&dc->sym, &old_sym, sizeof(CGrSym)); dc->color = old_color; dc->thick = old_pen_width; dc->flags = dc->flags & ~(DCF_SYMMETRY | DCF_TRANSFORMATION) | old_flags & (DCF_SYMMETRY | DCF_TRANSFORMATION); } public U0 Sprite3B(CDC *dc=gr.dc, I64 x, I64 y, I64 z, U8 *elems) {//Plot a sprite into a CDC, post transform xyz translation. I64 old_x = dc->x, old_y = dc->y, old_z = dc->z, old_flags = dc->flags & DCF_TRANSFORMATION; dc->x = x; dc->y = y; dc->z = z; dc->flags |= DCF_TRANSFORMATION; Sprite3(dc, 0, 0, 0, elems); dc->x = old_x; dc->y = old_y; dc->z = old_z; dc->flags = dc->flags & ~DCF_TRANSFORMATION | old_flags; } public U0 Sprite3Mat4x4B(CDC *dc=gr.dc, I64 x, I64 y, I64 z, U8 *elems, I64 *m) {//Plot rotated by matrix. I64 r[16], *old_r = dc->r, new_m[16], old_flags = dc->flags & DCF_TRANSFORMATION; MemCopy(new_m, m, 16 * sizeof(I64)); dc->flags |= DCF_TRANSFORMATION; Mat4x4TranslationAdd(new_m, x, y, z); dc->r=Mat4x4MulMat4x4Equ(r, old_r, new_m); Sprite3(dc, 0, 0, 0, elems); dc->r = old_r; dc->flags = dc->flags & ~DCF_TRANSFORMATION | old_flags; } public U0 Sprite3XB(CDC *dc=gr.dc, I64 x, I64 y, I64 z, U8 *elems, F64 phi=0) {//Plot rotated around X axis. I64 r[16]; Mat4x4IdentEqu(r); Mat4x4RotX(r, phi); Sprite3Mat4x4B(dc, x, y, z, elems, r); } public U0 Sprite3YB(CDC *dc=gr.dc, I64 x, I64 y, I64 z, U8 *elems, F64 omega=0) {//Plot rotated around Y axis. I64 r[16]; Mat4x4IdentEqu(r); Mat4x4RotY(r, omega); Sprite3Mat4x4B(dc, x, y, z, elems, r); } public U0 Sprite3ZB(CDC *dc=gr.dc, I64 x, I64 y, I64 z, U8 *elems, F64 theta=0) {//Plot rotated around Z axis. I64 r[16]; Mat4x4IdentEqu(r); Mat4x4RotZ(r, theta); Sprite3Mat4x4B(dc, x, y, z, elems, r); } public U0 SpriteExtents(U8 *elems, I64 *min_x=NULL, I64 *max_x=NULL, I64 *min_y=NULL, I64 *max_y=NULL) {//Ignores flood fills. CDC *dc = DCNew(I32_MAX, I32_MAX, Fs, TRUE); DCExtentsInit(dc); Sprite3(dc, I32_MAX / 2, I32_MAX / 2, I32_MAX / 2, elems); if (dc->min_x <= dc->max_x) { dc->min_x -= I32_MAX / 2; dc->max_x -= I32_MAX / 2; } if (dc->min_y <= dc->max_y) { dc->min_y -= I32_MAX / 2; dc->max_y -= I32_MAX / 2; } if (min_x) *min_x = dc->min_x; if (max_x) *max_x = dc->max_x; if (min_y) *min_y = dc->min_y; if (max_y) *max_y = dc->max_y; DCDel(dc); } public CDC *Sprite2DC(U8 *elems) {//Convert sprite to device context. CDC *res; I64 min_x, max_x, min_y, max_y; SpriteExtents(elems, &min_x, &max_x, &min_y, &max_y); res = DCNew(max_x - min_x + 1, max_y - min_y + 1); Sprite3(res, -min_x, -min_y, 0, elems); return res; } public U8 *SpriteInterpolate(F64 t, U8 *elems0, U8 *elems1) {//The two CSprite should be ident except for points shifted around. //t ranges from 0.0 to 1.0. I64 i, t1 = GR_SCALE * t, t0 = GR_SCALE - t1; I32 *ptr0, *ptr1, *ptrr; CD3I32 *p0, *p1, *pr; U8 *res; CSprite *tmpg0 = elems0 - offset(CSprite.start), *tmpg1 = elems1 - offset(CSprite.start), *tmpgr; if (t < 0.5) { i = SpriteSize(elems0), res = MAlloc(i); MemCopy(res, elems0, i); } else { i = SpriteSize(elems1), res = MAlloc(i); MemCopy(res, elems1, i); } tmpgr = res - offset(CSprite.start); while (tmpg0->type & SPG_TYPE_MASK) { if (tmpg0->type & SPG_TYPE_MASK != tmpg1->type & SPG_TYPE_MASK) throw('Graphics'); switch (tmpg0->type & SPG_TYPE_MASK) { case SPT_ROTATED_RECT: tmpgr->ppa.angle = (tmpg0->ppa.angle * t0 + tmpg1->ppa.angle * t1) / GR_SCALE; case SPT_RECT: case SPT_LINE: case SPT_ARROW: case SPT_PLANAR_SYMMETRY: tmpgr->pp.x2 = (tmpg0->pp.x2 * t0 + tmpg1->pp.x2 * t1) >> 32; tmpgr->pp.y2 = (tmpg0->pp.y2 * t0 + tmpg1->pp.y2 * t1) >> 32; case SPT_TEXT: case SPT_TEXT_BOX: case SPT_TEXT_DIAMOND: case SPT_PT: case SPT_FLOOD_FILL: case SPT_FLOOD_FILL_NOT: case SPT_SHIFT: tmpgr->p.x1 = (tmpg0->p.x1 * t0 + tmpg1->p.x1 * t1) >> 32; tmpgr->p.y1 = (tmpg0->p.y1 * t0 + tmpg1->p.y1 * t1) >> 32; break; case SPT_CIRCLE: tmpgr->pr.radius = (tmpg0->pr.radius * t0 + tmpg1->pr.radius * t1) >> 32; tmpgr->pr.x1 = (tmpg0->pr.x1 * t0 + tmpg1->pr.x1 * t1) >> 32; tmpgr->pr.y1 = (tmpg0->pr.y1 * t0 + tmpg1->pr.y1 * t1) >> 32; break; case SPT_ELLIPSE: case SPT_POLYGON: tmpgr->pwha.x1 = (tmpg0->pwha.x1 * t0 + tmpg1->pwha.x1 * t1) >> 32; tmpgr->pwha.y1 = (tmpg0->pwha.y1 * t0 + tmpg1->pwha.y1 * t1) >> 32; tmpgr->pwha.width = (tmpg0->pwha.width * t0 + tmpg1->pwha.width * t1) >> 32; tmpgr->pwha.height = (tmpg0->pwha.height * t0 + tmpg1->pwha.height * t1) >> 32; break; case SPT_BITMAP: tmpgr->pwhu.x1 = (tmpg0->pwhu.x1 * t0 + tmpg1->pwhu.x1 * t1) >> 32; tmpgr->pwhu.y1 = (tmpg0->pwhu.y1 * t0 + tmpg1->pwhu.y1 * t1) >> 32; break; case SPT_POLYLINE: ptr0 = &tmpg0->nu.u; ptr1 = &tmpg1->nu.u; ptrr = &tmpgr->nu.u; for (i = 0; i < tmpg0->nu.num; i++) { ptrr[i << 1] = (ptr0[i << 1] * t0 + ptr1[i << 1] * t1) >> 32; ptrr[i << 1 + 1] = (ptr0[i << 1 + 1] * t0 + ptr1[i << 1 + 1] * t1) >> 32; } break; case SPT_POLYPT: tmpgr->npu.x = (tmpg0->npu.x * t0 + tmpg1->npu.x * t1) >> 32; tmpgr->npu.y = (tmpg0->npu.y * t0 + tmpg1->npu.y * t1) >> 32; break; case SPT_BSPLINE2: case SPT_BSPLINE3: case SPT_BSPLINE2_CLOSED: case SPT_BSPLINE3_CLOSED: p0 = &tmpg0->nu.u; p1 = &tmpg1->nu.u; pr = &tmpgr->nu.u; for (i = 0; i < tmpg0->nu.num; i++) { pr[i].x = (p0[i].x * t0 + p1[i].x * t1) >> 32; pr[i].y = (p0[i].y * t0 + p1[i].y * t1) >> 32; pr[i].z = (p0[i].z * t0 + p1[i].z * t1) >> 32; } break; case SPT_MESH: p0 = &tmpg0->mu.u; p1 = &tmpg1->mu.u; pr = &tmpgr->mu.u; for (i = 0; i < tmpg0->mu.vertex_count; i++) { pr[i].x = (p0[i].x * t0 + p1[i].x * t1) >> 32; pr[i].y = (p0[i].y * t0 + p1[i].y * t1) >> 32; pr[i].z = (p0[i].z * t0 + p1[i].z * t1) >> 32; } break; case SPT_SHIFTABLE_MESH: p0 = &tmpg0->pmu.u; p1 = &tmpg1->pmu.u; pr = &tmpgr->pmu.u; for (i = 0; i < tmpg0->pmu.vertex_count; i++) { pr[i].x = (p0[i].x * t0 + p1[i].x * t1) >> 32; pr[i].y = (p0[i].y * t0 + p1[i].y * t1) >> 32; pr[i].z = (p0[i].z * t0 + p1[i].z * t1) >> 32; } break; } tmpg0(U8 *) += SpriteElemSize(tmpg0); tmpg1(U8 *) += SpriteElemSize(tmpg1); tmpgr(U8 *) += SpriteElemSize(tmpgr); } return res; } #help_index "Graphics/Sprite;DolDoc/Output;StdOut/DolDoc" public CDocEntry *DocSprite(CDoc *doc=NULL, U8 *elems, U8 *format=NULL) {//Put a sprite into a document. You can, optionally, supply a format string //for DolDoc cmd with a %d for the bin_num. I64 size; U8 *st; Bool unlock; CDocEntry *doc_e; CDocBin *tmpb; if (!doc && !(doc = DocPut)) return NULL; unlock = DocLock(doc); size = SpriteSize(elems); tmpb = CAlloc(sizeof(CDocBin), doc->mem_task); tmpb->size = size; tmpb->data = MAlloc(size, doc->mem_task); MemCopy(tmpb->data, elems, size); tmpb->num = doc->cur_bin_num; tmpb->use_count = 1; QueueInsert(tmpb, doc->bin_head.last); if (format) st = MStrPrint(format, doc->cur_bin_num++); else st = MStrPrint("$SP,\"\",BI=%d$", doc->cur_bin_num++); doc_e = DocPrint(doc, "%s", st); Free(st); doc_e->bin_data = tmpb; if (doc_e && doc_e->de_flags & DOCEF_TAG && doc_e->tag && *doc_e->tag) tmpb->tag = StrNew(doc_e->tag, doc->mem_task); if (unlock) DocUnlock(doc); return doc_e; } public CDocEntry *Sprite(U8 *elems, U8 *format=NULL) {//Put sprite to the command-line, DocPut. //If you set format, then include dollars ("$SP ...$") and leave %d for num. CDoc *doc; if (doc = DocPut) return DocSprite(doc, elems, format); return NULL; }