/* * COPYRIGHT * * pcb-rnd, interactive printed circuit board design * Copyright (C) 2017..2020 Tibor 'Igor2' Palinkas * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. * * Contact: * Project page: http://repo.hu/projects/pcb-rnd * lead developer: http://repo.hu/projects/pcb-rnd/contact.html * mailing list: pcb-rnd (at) list.repo.hu (send "subscribe") */ #include "config.h" #include #include #include #include #include "buffer.h" #include "board.h" #include "crosshair.h" #include "data.h" #include "data_it.h" #include #include "obj_subc.h" #include "obj_subc_op.h" #include "obj_subc_parent.h" #include "obj_poly_op.h" #include "obj_pstk.h" #include "obj_pstk_inlines.h" #include "obj_pstk_draw.h" #include "obj_line_op.h" #include "obj_term.h" #include "obj_text_draw.h" #include #include "draw.h" #include "draw_wireframe.h" #include "flag.h" #include "polygon.h" #include "operation.h" #include "undo.h" #include "move.h" #include #include #include "pcb_minuid.h" #include "conf_core.h" #include #include #include "extobj.h" static const char core_subc_cookie[] = "core-subc"; void pcb_subc_reg(pcb_data_t *data, pcb_subc_t *subc) { pcb_subclist_append(&data->subc, subc); pcb_obj_id_reg(data, subc); PCB_SET_PARENT(subc->data, subc, subc); PCB_SET_PARENT(subc, data, data); } void pcb_subc_unreg(pcb_subc_t *subc) { pcb_data_t *data = subc->parent.data; assert(subc->parent_type == PCB_PARENT_DATA); pcb_subclist_remove(subc); pcb_obj_id_del(data, subc); PCB_CLEAR_PARENT(subc); } /* Modify dst to include src, if src is not a floater */ static void pcb_box_bump_box_noflt(rnd_box_t *dst, rnd_box_t *src) { if (!PCB_FLAG_TEST(PCB_FLAG_FLOATER, ((pcb_any_obj_t *)(src)))) rnd_box_bump_box(dst, src); } /* update cached values: e.g. looking up refdes in attributes each time the netlist code needs it would be too expensive. Instead, we maintain a read-only ->refdes field and update it any time attributes change. */ static void pcb_subc_attrib_post_change(pcb_attribute_list_t *list, const char *name, const char *value) { pcb_subc_t *sc = (pcb_subc_t *)(((char *)list) - offsetof(pcb_subc_t, Attributes)); if (strcmp(name, "refdes") == 0) { const char *inv; sc->refdes = value; inv = pcb_obj_id_invalid(sc->refdes); if (inv != NULL) rnd_message(RND_MSG_ERROR, "Invalid character '%c' in subc refdes '%s'\n", *inv, sc->refdes); } else if (strcmp(name, "extobj") == 0) sc->extobj = value; pcb_text_dyn_bbox_update(sc->data); } pcb_subc_t *pcb_subc_alloc_id(long int id) { pcb_subc_t *sc; sc = calloc(sizeof(pcb_subc_t), 1); sc->ID = id; sc->Attributes.post_change = pcb_subc_attrib_post_change; sc->data = pcb_data_new(NULL); sc->type = PCB_OBJ_SUBC; PCB_SET_PARENT(sc->data, subc, sc); minuid_gen(&pcb_minuid, sc->uid); pcb_term_init(&sc->terminals); return sc; } pcb_subc_t *pcb_subc_alloc(void) { return pcb_subc_alloc_id(pcb_create_ID_get()); } pcb_subc_t *pcb_subc_new(void) { return pcb_subc_alloc(); } void pcb_subc_free(pcb_subc_t *sc) { pcb_extobj_del_pre(sc); if ((sc->parent.data != NULL) && (sc->parent.data->subc_tree != NULL)) rnd_rtree_delete(sc->parent.data->subc_tree, sc, (rnd_rtree_box_t *)sc); pcb_attribute_free(&sc->Attributes); if (sc->parent_type != PCB_PARENT_INVALID) pcb_subc_unreg(sc); pcb_data_free(sc->data); pcb_term_uninit(&sc->terminals); pcb_obj_common_free((pcb_any_obj_t *)sc); free(sc); } /* Create (and append) a new bound layer to a subc */ static pcb_layer_t *pcb_subc_layer_create_buff(pcb_subc_t *sc, pcb_layer_t *src) { pcb_layer_t *dst = &sc->data->Layer[sc->data->LayerN++]; memcpy(&dst->meta, &src->meta, sizeof(src->meta)); dst->is_bound = 1; dst->comb = src->comb; dst->parent.data = sc->data; dst->parent_type = PCB_PARENT_DATA; dst->type = PCB_OBJ_LAYER; dst->name = rnd_strdup(src->name); return dst; } pcb_layer_t *pcb_subc_layer_create(pcb_subc_t *sc, const char *name, pcb_layer_type_t type, pcb_layer_combining_t comb, int stack_offs, const char *purpose) { pcb_layer_t *dst = &sc->data->Layer[sc->data->LayerN++]; dst->is_bound = 1; dst->meta.bound.type = type; dst->comb = comb; dst->meta.bound.stack_offs = stack_offs; if (purpose != NULL) dst->meta.bound.purpose = rnd_strdup(purpose); else dst->meta.bound.purpose = NULL; dst->parent.data = sc->data; dst->parent_type = PCB_PARENT_DATA; dst->type = PCB_OBJ_LAYER; dst->name = rnd_strdup(name); return dst; } static pcb_line_t *add_aux_line(pcb_layer_t *aux, const char *key, const char *val, rnd_coord_t x1, rnd_coord_t y1, rnd_coord_t x2, rnd_coord_t y2) { pcb_line_t *l = pcb_line_new(aux, x1, y1, x2, y2, RND_MM_TO_COORD(0.1), 0, pcb_no_flags()); pcb_attribute_put(&l->Attributes, key, val); return l; } static pcb_line_t *find_aux_line(pcb_layer_t *aux, const char *key) { pcb_line_t *line; gdl_iterator_t it; linelist_foreach(&aux->Line, &it, line) { const char *val = pcb_attribute_get(&line->Attributes, "subc-role"); if ((val != NULL) && (strcmp(val, key) == 0)) return line; } return NULL; } /* Find and save the aux layer in the cache, of it exists */ void pcb_subc_cache_find_aux(pcb_subc_t *sc) { int n; for(n = 0; n < sc->data->LayerN; n++) { if (strcmp(sc->data->Layer[n].name, SUBC_AUX_NAME) == 0) { sc->aux_layer = sc->data->Layer + n; return; } } if (sc->aux_layer == NULL) sc->aux_layer = pcb_layer_new_bound(sc->data, PCB_LYT_VIRTUAL | PCB_LYT_NOEXPORT | PCB_LYT_MISC | PCB_LYT_TOP, SUBC_AUX_NAME, NULL); } /* Looking up aux objects for determining orientation/origin of a subc is somewhat expensive. Instead of doing that every time, we just save the pointer of those objects/layers. They don't change once the subc is loaded into memory */ static int pcb_subc_cache_update(pcb_subc_t *sc) { if (sc->aux_layer == NULL) pcb_subc_cache_find_aux(sc); if (sc->aux_cache[0] != NULL) return 0; if (sc->aux_layer == NULL) { rnd_message(RND_MSG_WARNING, "Can't find subc aux layer\n"); return -1; } sc->aux_cache[PCB_SUBCH_ORIGIN] = find_aux_line(sc->aux_layer, "origin"); sc->aux_cache[PCB_SUBCH_X] = find_aux_line(sc->aux_layer, "x"); sc->aux_cache[PCB_SUBCH_Y] = find_aux_line(sc->aux_layer, "y"); return 0; } rnd_bool pcb_subc_find_aux_point(pcb_subc_t *sc, const char *role, rnd_coord_t *x, rnd_coord_t *y) { pcb_line_t *l; if (sc->aux_layer == NULL) pcb_subc_cache_update(sc); if (sc->aux_layer == NULL) return rnd_false; l = find_aux_line(sc->aux_layer, role); if (l == NULL) return rnd_false; *x = l->Point1.X; *y = l->Point1.Y; return rnd_true; } int pcb_subc_get_origin(pcb_subc_t *sc, rnd_coord_t *x, rnd_coord_t *y) { if ((pcb_subc_cache_update(sc) != 0) || (sc->aux_cache[PCB_SUBCH_ORIGIN] == NULL)) return -1; *x = sc->aux_cache[PCB_SUBCH_ORIGIN]->Point1.X; *y = sc->aux_cache[PCB_SUBCH_ORIGIN]->Point1.Y; return 0; } int pcb_subc_get_rotation(pcb_subc_t *sc, double *rot) { double r, rr; if ((pcb_subc_cache_update(sc) != 0) || (sc->aux_cache[PCB_SUBCH_X] == NULL)) return -1; r = -1 * RND_RAD_TO_DEG * atan2(sc->aux_cache[PCB_SUBCH_X]->Point2.Y - sc->aux_cache[PCB_SUBCH_X]->Point1.Y, sc->aux_cache[PCB_SUBCH_X]->Point2.X - sc->aux_cache[PCB_SUBCH_X]->Point1.X); /* ugly hack to get round angles where possible: if error to a round angle is less than 1/10000, it was meant to be round, just got ruined by rounding errors */ rr = rnd_round(r*10000.0) / 10000.0; if (rr - r < 0.0001) *rot = rr; else *rot = r; return 0; } /* The subc is originally drawn on the top side, and the aux layer gets the PCB_LYT_TOP flag. If the subc is ever sent to the other side, the aux layer changed side too. This is to be detected here. */ int pcb_subc_get_side(pcb_subc_t *sc, int *on_bottom) { if ((pcb_subc_cache_update(sc) != 0) || (sc->aux_layer == NULL)) return -1; if (!sc->aux_layer->is_bound) return -1; if (sc->aux_layer->meta.bound.type & PCB_LYT_TOP) *on_bottom = 0; else if (sc->aux_layer->meta.bound.type & PCB_LYT_BOTTOM) *on_bottom = 1; else return -1; return 0; } int pcb_subc_get_host_trans(pcb_subc_t *sc, pcb_host_trans_t *tr, int neg) { int res = 0; double rr; if (pcb_subc_get_origin(sc, &tr->ox, &tr->oy) != 0) { tr->ox = tr->oy = 0; res = -1; } if (pcb_subc_get_rotation(sc, &tr->rot) != 0) { tr->rot = 0; res = -1; } if (pcb_subc_get_side(sc, &tr->on_bottom) != 0) { tr->on_bottom = 0; res = -1; } rr = tr->rot / RND_RAD_TO_DEG; if (neg) { tr->rot = -tr->rot; rr = -rr; tr->ox = -tr->ox; tr->oy = -tr->oy; } tr->cosa = cos(rr); tr->sina = sin(rr); return res; } static int pcb_subc_move_origin_(pcb_subc_t *sc, rnd_coord_t dx, rnd_coord_t dy, rnd_bool and_undo) { int n; if ((dx == 0) && (dy == 0)) return 0; for(n = 0; n < PCB_SUBCH_max; n++) { pcb_line_t *l = sc->aux_cache[n]; pcb_line_move(l, dx, dy); if (and_undo) pcb_undo_add_obj_to_move(PCB_OBJ_LINE, l->parent.layer, l, l, dx, dy); } return 0; } int pcb_subc_move_origin(pcb_subc_t *sc, rnd_coord_t dx, rnd_coord_t dy, rnd_bool and_undo) { if ((pcb_subc_cache_update(sc) != 0) || (sc->aux_cache[PCB_SUBCH_ORIGIN] == NULL)) return -1; return pcb_subc_move_origin_(sc, dx, dy, and_undo); } int pcb_subc_move_origin_to(pcb_subc_t *sc, rnd_coord_t x, rnd_coord_t y, rnd_bool and_undo) { rnd_coord_t ox, oy; if ((pcb_subc_cache_update(sc) != 0) || (sc->aux_cache[PCB_SUBCH_ORIGIN] == NULL)) return -1; ox = sc->aux_cache[PCB_SUBCH_ORIGIN]->Point1.X; oy = sc->aux_cache[PCB_SUBCH_ORIGIN]->Point1.Y; return pcb_subc_move_origin_(sc, x-ox, y-oy, and_undo); } static void pcb_subc_cache_invalidate(pcb_subc_t *sc) { sc->aux_cache[0] = NULL; } /* Convert a square line into a polygon and add it to dst */ static pcb_poly_t *sqline2term(pcb_layer_t *dst, pcb_line_t *line) { pcb_poly_t *poly; rnd_coord_t x[4], y[4]; int n; pcb_sqline_to_rect(line, x, y); poly = pcb_poly_new(dst, line->Clearance, pcb_no_flags()); for(n = 0; n < 4; n++) pcb_poly_point_new(poly, x[n], y[n]); PCB_FLAG_SET(PCB_FLAG_CLEARPOLYPOLY, poly); pcb_attribute_copy_all(&poly->Attributes, &line->Attributes); pcb_poly_init_clip(dst->parent.data, dst, poly); pcb_add_poly_on_layer(dst, poly); return poly; } pcb_layer_t *pcb_loose_subc_layer(pcb_board_t *pcb, pcb_layer_t *layer, rnd_bool alloc) { pcb_subc_t *sc; int n; if (!pcb->is_footprint) return layer; sc = pcb_subclist_first(&pcb->Data->subc); if (sc == NULL) return layer; for(n = 0; n < sc->data->LayerN; n++) { pcb_layer_t *l = &sc->data->Layer[n]; if (!l->is_bound) continue; if (l->meta.bound.real == layer) return l; } /* the subc does not have that layer */ if (alloc) { pcb_layer_t *nlayer = &sc->data->Layer[sc->data->LayerN]; sc->data->LayerN++; pcb_layer_real2bound(nlayer, layer, 1); return nlayer; } return layer; } void pcb_subc_as_board_update(pcb_board_t *pcb) { pcb_subc_t *sc; if (!pcb->is_footprint) return; sc = pcb_subclist_first(&pcb->Data->subc); if (sc == NULL) return; pcb_subc_bbox(sc); } static rnd_coord_t read_mask(pcb_any_obj_t *obj) { const char *smask = pcb_attribute_get(&obj->Attributes, "elem_smash_pad_mask"); rnd_coord_t mask = 0; if (smask != NULL) { rnd_bool success; mask = rnd_get_value_ex(smask, NULL, NULL, NULL, "mm", &success); if (!success) mask = 0; } return mask; } static void get_aux_layer(pcb_subc_t *sc, int alloc) { if (sc->aux_layer == NULL) pcb_subc_cache_find_aux(sc); pcb_subc_cache_update(sc); } void pcb_subc_create_aux(pcb_subc_t *sc, rnd_coord_t ox, rnd_coord_t oy, double rot, rnd_bool bottom) { double unit = PCB_SUBC_AUX_UNIT; double cs, sn, ydir = 1; get_aux_layer(sc, 1); if (rot == 0.0) { cs = 1; sn = 0; } else { cs = cos(rot/RND_RAD_TO_DEG); sn = sin(rot/RND_RAD_TO_DEG); } if (bottom) { assert(sc->aux_layer->is_bound); sc->aux_layer->meta.bound.type &= ~PCB_LYT_TOP; sc->aux_layer->meta.bound.type |= PCB_LYT_BOTTOM; ydir = -1; } add_aux_line(sc->aux_layer, "subc-role", "origin", ox, oy, ox, oy); add_aux_line(sc->aux_layer, "subc-role", "x", ox, oy, rnd_round((double)ox + cs*unit), rnd_round((double)oy + sn*unit)); add_aux_line(sc->aux_layer, "subc-role", "y", ox, oy, rnd_round((double)ox + ydir*sn*unit), rnd_round((double)oy + ydir*cs*unit)); } void pcb_subc_create_aux_point(pcb_subc_t *sc, rnd_coord_t x, rnd_coord_t y, const char *role) { get_aux_layer(sc, 1); add_aux_line(sc->aux_layer, "subc-role", role, x, y, x, y); } int pcb_subc_convert_from_buffer(pcb_buffer_t *buffer) { pcb_subc_t *sc; int n, top_pads = 0, bottom_pads = 0, has_refdes_text = 0; pcb_layer_t *dst_top_mask = NULL, *dst_bottom_mask = NULL, *dst_top_paste = NULL, *dst_bottom_paste = NULL, *dst_top_silk = NULL; vtp0_t mask_pads, paste_pads; vtp0_init(&mask_pads); vtp0_init(&paste_pads); sc = pcb_subc_alloc(); pcb_subc_reg(buffer->Data, sc); /* create layer matches and copy objects */ for(n = 0; n < PCB_MAX_LAYER; n++) { pcb_layer_t *dst, *src; pcb_line_t *line; pcb_text_t *text; pcb_poly_t *poly; pcb_arc_t *arc; pcb_gfx_t *gfx; pcb_layer_type_t ltype; if (pcb_layer_is_pure_empty(&buffer->Data->Layer[n])) continue; src = &buffer->Data->Layer[n]; dst = pcb_subc_layer_create_buff(sc, src); ltype = dst->meta.bound.type; if ((dst->comb & PCB_LYC_SUB) == 0) { if ((ltype & PCB_LYT_PASTE) && (ltype & PCB_LYT_TOP)) dst_top_paste = dst; else if ((ltype & PCB_LYT_PASTE) && (ltype & PCB_LYT_BOTTOM)) dst_bottom_paste = dst; else if ((ltype & PCB_LYT_SILK) && (ltype & PCB_LYT_TOP)) dst_top_silk = dst; } if (dst->comb & PCB_LYC_SUB) { if ((ltype & PCB_LYT_MASK) && (ltype & PCB_LYT_TOP)) dst_top_mask = dst; else if ((ltype & PCB_LYT_MASK) && (ltype & PCB_LYT_BOTTOM)) dst_bottom_mask = dst; } while((line = linelist_first(&src->Line)) != NULL) { rnd_coord_t mask = 0; char *np, *sq, *termpad; const char *term; termpad = pcb_attribute_get(&line->Attributes, "elem_smash_pad"); if (termpad != NULL) { if (ltype & PCB_LYT_TOP) top_pads++; else if (ltype & PCB_LYT_BOTTOM) bottom_pads++; mask = read_mask((pcb_any_obj_t *)line); } term = pcb_attribute_get(&line->Attributes, "term"); np = pcb_attribute_get(&line->Attributes, "elem_smash_nopaste"); sq = pcb_attribute_get(&line->Attributes, "elem_smash_shape_square"); if ((sq != NULL) && (*sq == '1')) { /* convert to polygon */ poly = sqline2term(dst, line); if (termpad != NULL) { if ((np == NULL) || (*np != '1')) { poly = sqline2term(dst, line); vtp0_append(&paste_pads, poly); } if (mask > 0) { line->Thickness = mask; poly = sqline2term(dst, line); if (term != NULL) pcb_attribute_put(&poly->Attributes, "term", term); vtp0_append(&mask_pads, poly); } } pcb_line_free(line); } else { /* copy the line */ pcb_line_unreg(line); pcb_line_reg(dst, line); PCB_FLAG_CLEAR(PCB_FLAG_WARN | PCB_FLAG_FOUND | PCB_FLAG_SELECTED, line); if (termpad != NULL) { pcb_line_t *nl = pcb_line_dup(dst, line); if ((np == NULL) || (*np != '1')) vtp0_append(&paste_pads, nl); if (mask > 0) { nl = pcb_line_dup(dst, line); nl->Thickness = mask; if (term != NULL) pcb_attribute_put(&nl->Attributes, "term", term); vtp0_append(&mask_pads, nl); } } } } while((arc = arclist_first(&src->Arc)) != NULL) { pcb_arc_unreg(arc); pcb_arc_reg(dst, arc); PCB_FLAG_CLEAR(PCB_FLAG_WARN | PCB_FLAG_FOUND | PCB_FLAG_SELECTED, arc); } while((gfx = gfxlist_first(&src->Gfx)) != NULL) { pcb_gfx_unreg(gfx); pcb_gfx_reg(dst, gfx); PCB_FLAG_CLEAR(PCB_FLAG_WARN | PCB_FLAG_FOUND | PCB_FLAG_SELECTED, gfx); } while((text = textlist_first(&src->Text)) != NULL) { pcb_text_unreg(text); pcb_text_reg(dst, text); PCB_FLAG_CLEAR(PCB_FLAG_WARN | PCB_FLAG_FOUND | PCB_FLAG_SELECTED, text); if (!has_refdes_text && (text->TextString != NULL) && (strstr(text->TextString, "%a.parent.refdes%") != NULL)) has_refdes_text = 1; } while((poly = polylist_first(&src->Polygon)) != NULL) { pcb_poly_unreg(poly); pcb_poly_reg(dst, poly); PCB_FLAG_CLEAR(PCB_FLAG_WARN | PCB_FLAG_FOUND | PCB_FLAG_SELECTED, poly); } } /* create paste and mask side effects - needed when importing from footprint */ { if (top_pads > 0) { if (dst_top_paste == NULL) dst_top_paste = pcb_layer_new_bound(sc->data, PCB_LYT_TOP | PCB_LYT_PASTE, "top paste", NULL); if (dst_top_mask == NULL) { dst_top_mask = pcb_layer_new_bound(sc->data, PCB_LYT_TOP | PCB_LYT_MASK, "top mask", NULL); dst_top_mask->comb = PCB_LYC_SUB; } } if (bottom_pads > 0) { if (dst_bottom_paste == NULL) dst_bottom_paste = pcb_layer_new_bound(sc->data, PCB_LYT_BOTTOM | PCB_LYT_PASTE, "bottom paste", NULL); if (dst_bottom_mask == NULL) { dst_bottom_mask = pcb_layer_new_bound(sc->data, PCB_LYT_BOTTOM | PCB_LYT_MASK, "bottom mask", NULL); dst_bottom_mask->comb = PCB_LYC_SUB; } } } { /* convert globals */ pcb_pstk_t *ps; while((ps = padstacklist_first(&buffer->Data->padstack)) != NULL) { const pcb_pstk_proto_t *proto = pcb_pstk_get_proto(ps); pcb_pstk_unreg(ps); pcb_pstk_reg(sc->data, ps); PCB_FLAG_CLEAR(PCB_FLAG_WARN | PCB_FLAG_FOUND | PCB_FLAG_SELECTED, ps); ps->proto = pcb_pstk_proto_insert_dup(sc->data, proto, 1, 0); ps->protoi = -1; } } vtp0_uninit(&mask_pads); vtp0_uninit(&paste_pads); pcb_subc_create_aux(sc, buffer->X, buffer->Y, 0.0, !!buffer->from_bottom_side); /* Add refdes */ if ((conf_core.editor.subc_conv_refdes != NULL) && (*conf_core.editor.subc_conv_refdes != '\0')) { if (strcmp(conf_core.editor.subc_conv_refdes, "") != 0) pcb_attribute_put(&sc->Attributes, "refdes", conf_core.editor.subc_conv_refdes); if (!has_refdes_text) { pcb_layer_t *dstly; int botflg = 0; if (!buffer->from_bottom_side) { if (dst_top_silk == NULL) dst_top_silk = pcb_layer_new_bound(sc->data, PCB_LYT_TOP | PCB_LYT_SILK, "top-silk", NULL); dstly = dst_top_silk; } else { dstly = pcb_layer_new_bound(sc->data, PCB_LYT_BOTTOM | PCB_LYT_SILK, "bottom-silk", NULL); botflg = PCB_FLAG_ONSOLDER; } if (dstly != NULL) pcb_text_new(dstly, pcb_font(PCB, 0, 0), buffer->X, buffer->Y, 0, 100, 0, "%a.parent.refdes%", pcb_flag_make(PCB_FLAG_DYNTEXT | PCB_FLAG_FLOATER | botflg)); else rnd_message(RND_MSG_ERROR, "Error: can't create primary silk layer in subc for placing the refdes\n"); } } return 0; } static void pcb_subc_draw_origin(rnd_hid_gc_t GC, pcb_subc_t *sc, rnd_coord_t DX, rnd_coord_t DY) { pcb_line_t *origin; pcb_subc_cache_update(sc); origin = sc->aux_cache[PCB_SUBCH_ORIGIN]; if (origin == NULL) return; DX += (origin->Point1.X + origin->Point2.X) / 2; DY += (origin->Point1.Y + origin->Point2.Y) / 2; rnd_render->draw_line(GC, DX - PCB_EMARK_SIZE, DY, DX, DY - PCB_EMARK_SIZE); rnd_render->draw_line(GC, DX + PCB_EMARK_SIZE, DY, DX, DY - PCB_EMARK_SIZE); rnd_render->draw_line(GC, DX - PCB_EMARK_SIZE, DY, DX, DY + PCB_EMARK_SIZE); rnd_render->draw_line(GC, DX + PCB_EMARK_SIZE, DY, DX, DY + PCB_EMARK_SIZE); if (sc->auto_termname_display) { #define OFX (PCB_EMARK_SIZE/5) rnd_render->draw_line(GC, DX-OFX, DY-PCB_EMARK_SIZE/3, DX-OFX, DY+PCB_EMARK_SIZE/3); rnd_render->draw_line(GC, DX-OFX, DY+PCB_EMARK_SIZE/3, DX-OFX+PCB_EMARK_SIZE/3, DY+PCB_EMARK_SIZE/3); #undef OFX } } /* Draw a small lock on the bottom right corner at lx;ly */ static void pcb_subc_draw_locked(rnd_hid_gc_t GC, pcb_subc_t *sc, rnd_coord_t lx, rnd_coord_t ly, int on_bottom) { rnd_coord_t s = on_bottom ? -PCB_EMARK_SIZE : PCB_EMARK_SIZE; if (rnd_conf.editor.view.flip_x) lx += PCB_EMARK_SIZE*1.5; else lx -= PCB_EMARK_SIZE*1.5; if (on_bottom) ly += PCB_EMARK_SIZE*1.5; else ly -= PCB_EMARK_SIZE*1.5; rnd_render->draw_line(GC, lx-PCB_EMARK_SIZE, ly+PCB_EMARK_SIZE, lx+PCB_EMARK_SIZE, ly+PCB_EMARK_SIZE); rnd_render->draw_line(GC, lx-PCB_EMARK_SIZE, ly-PCB_EMARK_SIZE, lx+PCB_EMARK_SIZE, ly-PCB_EMARK_SIZE); rnd_render->draw_line(GC, lx-PCB_EMARK_SIZE, ly-PCB_EMARK_SIZE, lx-PCB_EMARK_SIZE, ly+PCB_EMARK_SIZE); rnd_render->draw_line(GC, lx+PCB_EMARK_SIZE, ly-PCB_EMARK_SIZE, lx+PCB_EMARK_SIZE, ly+PCB_EMARK_SIZE); rnd_render->draw_line(GC, lx-PCB_EMARK_SIZE, ly+PCB_EMARK_SIZE*1/3, lx+PCB_EMARK_SIZE, ly+PCB_EMARK_SIZE*1/3); rnd_render->draw_line(GC, lx-PCB_EMARK_SIZE, ly-PCB_EMARK_SIZE*1/3, lx+PCB_EMARK_SIZE, ly-PCB_EMARK_SIZE*1/3); rnd_render->draw_arc(GC, lx, ly-s, PCB_EMARK_SIZE*2/3, PCB_EMARK_SIZE*2/3, 180, on_bottom ? -180 : 180); } void pcb_xordraw_subc(pcb_subc_t *sc, rnd_coord_t DX, rnd_coord_t DY, int use_curr_side) { int n, mirr; rnd_coord_t w = 0, h = 0; mirr = use_curr_side && conf_core.editor.show_solder_side; /* mirror center */ if (mirr) { pcb_subc_get_origin(sc, &w, &h); w *= 2; h *= 2; } /* draw per layer objects */ for(n = 0; n < sc->data->LayerN; n++) { pcb_layer_t *ly = sc->data->Layer + n; pcb_line_t *line; pcb_text_t *text; pcb_poly_t *poly; pcb_arc_t *arc; pcb_gfx_t *gfx; gdl_iterator_t it; linelist_foreach(&ly->Line, &it, line) pcb_draw_wireframe_line(pcb_crosshair.GC, DX + PCB_CSWAP_X(line->Point1.X, w, mirr), DY + PCB_CSWAP_Y(line->Point1.Y, h, mirr), DX + PCB_CSWAP_X(line->Point2.X, w, mirr), DY + PCB_CSWAP_Y(line->Point2.Y, h, mirr), line->Thickness, 0 ); arclist_foreach(&ly->Arc, &it, arc) { if(arc->Width != arc->Height) { TODO(": The wireframe arc drawing code cannot draw ellipses yet so draw the elliptical arc with a thin line ") double sa = mirr ? RND_SWAP_ANGLE(arc->StartAngle) : arc->StartAngle; double da = mirr ? RND_SWAP_DELTA(arc->Delta) : arc->Delta; rnd_render->draw_arc(pcb_crosshair.GC, DX + PCB_CSWAP_X(arc->X, w, mirr), DY + PCB_CSWAP_Y(arc->Y, h, mirr), arc->Width, arc->Height, sa, da); } else { pcb_arc_t temp_arc; temp_arc.StartAngle = mirr ? RND_SWAP_ANGLE(arc->StartAngle) : arc->StartAngle; temp_arc.Delta = mirr ? RND_SWAP_DELTA(arc->Delta) : arc->Delta; temp_arc.X = DX + PCB_CSWAP_X(arc->X, w, mirr); temp_arc.Y = DY + PCB_CSWAP_Y(arc->Y, h, mirr); temp_arc.Width = arc->Width; temp_arc.Height = arc->Height; temp_arc.Thickness = arc->Thickness; pcb_draw_wireframe_arc(pcb_crosshair.GC, &temp_arc, arc->Thickness); } } gfxlist_foreach(&ly->Gfx, &it, gfx) { rnd_render->draw_line(pcb_crosshair.GC, DX+gfx->corner[0].X, DY+gfx->corner[0].Y, DX+gfx->corner[1].X, DY+gfx->corner[1].Y); rnd_render->draw_line(pcb_crosshair.GC, DX+gfx->corner[1].X, DY+gfx->corner[1].Y, DX+gfx->corner[2].X, DY+gfx->corner[2].Y); rnd_render->draw_line(pcb_crosshair.GC, DX+gfx->corner[2].X, DY+gfx->corner[2].Y, DX+gfx->corner[3].X, DY+gfx->corner[3].Y); rnd_render->draw_line(pcb_crosshair.GC, DX+gfx->corner[3].X, DY+gfx->corner[3].Y, DX+gfx->corner[0].X, DY+gfx->corner[0].Y); } polylist_foreach(&ly->Polygon, &it, poly) pcb_xordraw_poly_subc(poly, DX, DY, w, h, mirr); textlist_foreach(&ly->Text, &it, text) { if (mirr) { pcb_text_t t = {0}; t = *text; t.X = PCB_CSWAP_X(text->X, w, mirr); t.Y = PCB_CSWAP_Y(text->Y, h, mirr); t.BoundingBox.X1 = PCB_CSWAP_X(text->BoundingBox.X1, w, mirr); t.BoundingBox.Y1 = PCB_CSWAP_Y(text->BoundingBox.Y1, h, mirr); t.BoundingBox.X2 = PCB_CSWAP_X(text->BoundingBox.X2, w, mirr); t.BoundingBox.Y2 = PCB_CSWAP_Y(text->BoundingBox.Y2, h, mirr); PCB_FLAG_TOGGLE(PCB_FLAG_ONSOLDER, &t); pcb_text_draw_xor(&t, DX, DY, 1); } else pcb_text_draw_xor(text, DX, DY, 1); } } /* draw global objects */ { pcb_pstk_t *ps; gdl_iterator_t it; padstacklist_foreach(&sc->data->padstack, &it, ps) { rnd_coord_t ox, oy; int oxm, pri; ox = ps->x; oy = ps->y; oxm = ps->xmirror; pri = ps->protoi; ps->x = PCB_CSWAP_X(ps->x, w, mirr); ps->y = PCB_CSWAP_Y(ps->y, h, mirr); ps->x += DX; ps->y += DY; if (mirr) { ps->xmirror = !ps->xmirror; ps->protoi = -1; } pcb_pstk_thindraw(NULL, pcb_crosshair.GC, ps); ps->x = ox; ps->y = oy; ps->xmirror = oxm; ps->protoi = pri; } } pcb_subc_draw_origin(pcb_crosshair.GC, sc, DX, DY); } pcb_layer_t *pcb_subc_alloc_layer_like(pcb_subc_t *subc, pcb_layer_t *sl) { pcb_layer_t *dl; if (subc->data->LayerN >= PCB_MAX_LAYER) return NULL; dl = &subc->data->Layer[subc->data->LayerN++]; dl->is_bound = 1; dl->type = PCB_OBJ_LAYER; if (!sl->is_bound) { pcb_layer_t tmply; memset(&dl->meta.bound, 0, sizeof(sl->meta.bound)); pcb_layer_real2bound(&tmply, sl, 0); dl->meta.bound.type = tmply.meta.bound.type; dl->meta.bound.stack_offs = tmply.meta.bound.stack_offs; dl->meta.bound.purpose = tmply.meta.bound.purpose; } else memcpy(&dl->meta.bound, &sl->meta.bound, sizeof(sl->meta.bound)); dl->name = rnd_strdup(sl->name); dl->comb = sl->comb; if (dl->meta.bound.real != NULL) pcb_layer_link_trees(dl, dl->meta.bound.real); return dl; } #define MAYBE_KEEP_ID(dst, src) \ do { \ if ((keep_ids) && (dst != NULL)) \ dst->ID = src->ID; \ } while(0) pcb_subc_t *pcb_subc_copy_meta(pcb_subc_t *dst, const pcb_subc_t *src) { if (dst == NULL) return NULL; pcb_attribute_copy_all(&dst->Attributes, &src->Attributes); return dst; } void pcb_subc_dup_layer_objs(pcb_subc_t *dst_sc, pcb_layer_t *dl, pcb_layer_t *sl, rnd_coord_t dx, rnd_coord_t dy, rnd_bool keep_ids) { pcb_line_t *line, *nline; pcb_text_t *text, *ntext; pcb_arc_t *arc, *narc; pcb_gfx_t *gfx, *ngfx; gdl_iterator_t it; pcb_board_t *pcb = NULL; pcb_subc_t *src_sc = sl->parent.data->parent.subc; assert(dst_sc->parent_type == PCB_PARENT_DATA); if (dst_sc->parent.data->parent_type != PCB_PARENT_INVALID) { assert(dst_sc->parent.data->parent_type == PCB_PARENT_BOARD); pcb = dst_sc->parent.data->parent.board; } if (pcb != NULL) pcb_data_clip_inhibit_inc(pcb->Data); if (src_sc->aux_layer == NULL) pcb_subc_cache_find_aux(src_sc); linelist_foreach(&sl->Line, &it, line) { nline = pcb_line_dup_at(dl, line, dx, dy); MAYBE_KEEP_ID(nline, line); if (nline != NULL) { PCB_SET_PARENT(nline, layer, dl); if (!PCB_FLAG_TEST(PCB_FLAG_FLOATER, nline) && (sl != src_sc->aux_layer)) { pcb_box_bump_box_noflt(&dst_sc->BoundingBox, &nline->BoundingBox); pcb_box_bump_box_noflt(&dst_sc->bbox_naked, &nline->bbox_naked); } if (pcb != NULL) pcb_poly_clear_from_poly(pcb->Data, PCB_OBJ_LINE, dl, nline); } } arclist_foreach(&sl->Arc, &it, arc) { narc = pcb_arc_dup_at(dl, arc, dx, dy); MAYBE_KEEP_ID(narc, arc); if (narc != NULL) { PCB_SET_PARENT(narc, layer, dl); if (!PCB_FLAG_TEST(PCB_FLAG_FLOATER, narc)) { pcb_box_bump_box_noflt(&dst_sc->BoundingBox, &narc->BoundingBox); pcb_box_bump_box_noflt(&dst_sc->bbox_naked, &narc->bbox_naked); } if (pcb != NULL) pcb_poly_clear_from_poly(pcb->Data, PCB_OBJ_ARC, dl, narc); } } gfxlist_foreach(&sl->Gfx, &it, gfx) { ngfx = pcb_gfx_dup_at(dl, gfx, dx, dy); MAYBE_KEEP_ID(ngfx, gfx); if (ngfx != NULL) { PCB_SET_PARENT(ngfx, layer, dl); if (!PCB_FLAG_TEST(PCB_FLAG_FLOATER, ngfx)) { pcb_box_bump_box_noflt(&dst_sc->BoundingBox, &ngfx->BoundingBox); pcb_box_bump_box_noflt(&dst_sc->bbox_naked, &ngfx->bbox_naked); } if (pcb != NULL) pcb_poly_clear_from_poly(pcb->Data, PCB_OBJ_GFX, dl, ngfx); } } textlist_foreach(&sl->Text, &it, text) { ntext = pcb_text_dup_at(dl, text, dx, dy); MAYBE_KEEP_ID(ntext, text); if (ntext != NULL) { PCB_SET_PARENT(ntext, layer, dl); if (!PCB_FLAG_TEST(PCB_FLAG_FLOATER, ntext)) { pcb_box_bump_box_noflt(&dst_sc->BoundingBox, &ntext->BoundingBox); pcb_box_bump_box_noflt(&dst_sc->bbox_naked, &ntext->bbox_naked); } if (pcb != NULL) pcb_poly_clear_from_poly(pcb->Data, PCB_OBJ_TEXT, dl, ntext); } } if (pcb != NULL) pcb_data_clip_inhibit_dec(pcb->Data, 0); } pcb_subc_t *pcb_subc_dup_at(pcb_board_t *pcb, pcb_data_t *dst, const pcb_subc_t *src, rnd_coord_t dx, rnd_coord_t dy, rnd_bool keep_ids, rnd_bool undoable) { pcb_board_t *src_pcb; int n; pcb_subc_t *sc = pcb_subc_alloc(); int undoable_psproto = (undoable && (dst->parent_type == PCB_PARENT_BOARD)); if (keep_ids) sc->ID = src->ID; else sc->ID = pcb_create_ID_get(); sc->copied_from_ID = src->ID; minuid_cpy(sc->uid, src->uid); pcb_subc_reg(dst, sc); src_pcb = pcb_data_get_top(src->data); pcb_subc_copy_meta(sc, src); sc->BoundingBox.X1 = sc->BoundingBox.Y1 = RND_MAX_COORD; sc->BoundingBox.X2 = sc->BoundingBox.Y2 = -RND_MAX_COORD; sc->bbox_naked.X1 = sc->bbox_naked.Y1 = RND_MAX_COORD; sc->bbox_naked.X2 = sc->bbox_naked.Y2 = -RND_MAX_COORD; /* make a copy of layer data */ for(n = 0; n < src->data->LayerN; n++) { pcb_layer_t *sl = src->data->Layer + n; pcb_layer_t *dl = sc->data->Layer + n; /* bind layer/resolve layers */ dl->is_bound = 1; dl->type = PCB_OBJ_LAYER; if ((pcb != NULL) && (pcb == src_pcb)) { /* copy within the same board */ memcpy(&dl->meta.bound, &sl->meta.bound, sizeof(sl->meta.bound)); dl->name = rnd_strdup(sl->name); dl->comb = sl->comb; if (dl->meta.bound.real != NULL) pcb_layer_link_trees(dl, dl->meta.bound.real); } else if (pcb != NULL) { /* copying from buffer to board */ memcpy(&dl->meta.bound, &sl->meta.bound, sizeof(sl->meta.bound)); dl->name = rnd_strdup(sl->name); dl->meta.bound.real = pcb_layer_resolve_binding(pcb, sl); dl->comb = sl->comb; if (dl->meta.bound.real == NULL) { if (!(dl->meta.bound.type & PCB_LYT_VIRTUAL)) { const char *name = dl->name; if (name == NULL) name = ""; rnd_message(RND_MSG_WARNING, "Couldn't bind layer %s of subcircuit while placing it\n", name); } } else pcb_layer_link_trees(dl, dl->meta.bound.real); } else { /* copying from board to buffer */ memcpy(&dl->meta.bound, &sl->meta.bound, sizeof(sl->meta.bound)); dl->meta.bound.real = NULL; dl->name = rnd_strdup(sl->name); dl->comb = sl->comb; } pcb_subc_dup_layer_objs(sc, dl, sl, dx, dy, keep_ids); } sc->data->LayerN = src->data->LayerN; /* bind the via rtree so that vias added in this subc show up on the board */ if (pcb != NULL) { if (pcb->Data->padstack_tree == NULL) rnd_rtree_init(pcb->Data->padstack_tree = malloc(sizeof(rnd_rtree_t))); sc->data->padstack_tree = pcb->Data->padstack_tree; } { /* make a copy of global data */ pcb_pstk_t *ps, *nps; gdl_iterator_t it; padstacklist_foreach(&src->data->padstack, &it, ps) { rnd_cardinal_t pid = pcb_pstk_proto_insert_dup(sc->data, pcb_pstk_get_proto(ps), 1, undoable_psproto); nps = pcb_pstk_new_tr(sc->data, -1, pid, ps->x+dx, ps->y+dy, ps->Clearance, ps->Flags, ps->rot, ps->xmirror, ps->smirror); pcb_pstk_copy_meta(nps, ps); MAYBE_KEEP_ID(nps, ps); if (nps != NULL) { if (!PCB_FLAG_TEST(PCB_FLAG_FLOATER, nps)) { pcb_box_bump_box_noflt(&sc->BoundingBox, &nps->BoundingBox); pcb_box_bump_box_noflt(&sc->bbox_naked, &nps->bbox_naked); } } } } /* make a copy of polygons at the end so clipping caused by other objects are calculated only once */ for(n = 0; n < src->data->LayerN; n++) { pcb_layer_t *sl = src->data->Layer + n; pcb_layer_t *dl = sc->data->Layer + n; pcb_poly_t *poly, *npoly; gdl_iterator_t it; polylist_foreach(&sl->Polygon, &it, poly) { npoly = pcb_poly_dup_at(dl, poly, dx, dy); MAYBE_KEEP_ID(npoly, poly); if (npoly != NULL) { PCB_SET_PARENT(npoly, layer, dl); if (!PCB_FLAG_TEST(PCB_FLAG_FLOATER, npoly)) { pcb_box_bump_box_noflt(&sc->BoundingBox, &npoly->BoundingBox); pcb_box_bump_box_noflt(&sc->bbox_naked, &npoly->bbox_naked); } pcb_poly_ppclear(npoly); } } } memcpy(&sc->Flags, &src->Flags, sizeof(sc->Flags)); rnd_close_box(&sc->BoundingBox); rnd_close_box(&sc->bbox_naked); if (pcb != NULL) { if (!dst->subc_tree) rnd_rtree_init(dst->subc_tree = malloc(sizeof(rnd_rtree_t))); rnd_rtree_insert(dst->subc_tree, sc, (rnd_rtree_box_t *)sc); } return sc; } pcb_subc_t *pcb_subc_dup(pcb_board_t *pcb, pcb_data_t *dst, pcb_subc_t *src) { return pcb_subc_dup_at(pcb, dst, src, 0, 0, rnd_false, rnd_true); } void pcb_subc_bbox(pcb_subc_t *sc) { pcb_data_bbox(&sc->BoundingBox, sc->data, rnd_true); pcb_data_bbox_naked(&sc->bbox_naked, sc->data, rnd_true); } /* erases a subc on a layer */ void EraseSubc(pcb_subc_t *sc) { pcb_draw_invalidate(sc); } void DrawSubc(pcb_subc_t *sc) { pcb_draw_invalidate(sc); } /* Execute an operation on all children on an sc and update the bbox of the sc */ void *pcb_subc_op(pcb_data_t *Data, pcb_subc_t *sc, pcb_opfunc_t *opfunc, pcb_opctx_t *ctx, pcb_subc_op_undo_t batch_undo) { int n; if (Data != NULL) pcb_data_clip_inhibit_inc(Data); pcb_subc_part_changed_inhibit_inc(sc); switch(batch_undo) { case PCB_SUBCOP_UNDO_SUBC: pcb_undo_freeze_serial(); pcb_undo_freeze_add(); break; case PCB_SUBCOP_UNDO_BATCH: pcb_undo_inc_serial(); pcb_undo_save_serial(); pcb_undo_freeze_serial(); break; case PCB_SUBCOP_UNDO_NORMAL: break; } EraseSubc(sc); if (pcb_data_get_top(Data) != NULL) { if (Data->subc_tree != NULL) rnd_rtree_delete(Data->subc_tree, sc, (rnd_rtree_box_t *)sc); else rnd_rtree_init(Data->subc_tree = malloc(sizeof(rnd_rtree_t))); } /* for calculating the new bounding box on the fly */ sc->BoundingBox.X1 = sc->BoundingBox.Y1 = RND_MAX_COORD; sc->BoundingBox.X2 = sc->BoundingBox.Y2 = -RND_MAX_COORD; sc->bbox_naked.X1 = sc->bbox_naked.Y1 = RND_MAX_COORD; sc->bbox_naked.X2 = sc->bbox_naked.Y2 = -RND_MAX_COORD; if (sc->aux_layer == NULL) pcb_subc_cache_find_aux(sc); /* execute on layer locals */ for(n = 0; n < sc->data->LayerN; n++) { pcb_layer_t *sl = sc->data->Layer + n; pcb_line_t *line; pcb_text_t *text; pcb_poly_t *poly; pcb_arc_t *arc; pcb_gfx_t *gfx; gdl_iterator_t it; linelist_foreach(&sl->Line, &it, line) { pcb_object_operation(opfunc, ctx, PCB_OBJ_LINE, sl, line, line); if (!PCB_FLAG_TEST(PCB_FLAG_FLOATER, line) && (sl != sc->aux_layer)) { pcb_box_bump_box_noflt(&sc->BoundingBox, &line->BoundingBox); pcb_box_bump_box_noflt(&sc->bbox_naked, &line->bbox_naked); } } arclist_foreach(&sl->Arc, &it, arc) { pcb_object_operation(opfunc, ctx, PCB_OBJ_ARC, sl, arc, arc); if (!PCB_FLAG_TEST(PCB_FLAG_FLOATER, arc)) { pcb_box_bump_box_noflt(&sc->BoundingBox, &arc->BoundingBox); pcb_box_bump_box_noflt(&sc->bbox_naked, &arc->bbox_naked); } } textlist_foreach(&sl->Text, &it, text) { pcb_object_operation(opfunc, ctx, PCB_OBJ_TEXT, sl, text, text); if (!PCB_FLAG_TEST(PCB_FLAG_FLOATER, text)) { pcb_box_bump_box_noflt(&sc->BoundingBox, &text->BoundingBox); pcb_box_bump_box_noflt(&sc->bbox_naked, &text->bbox_naked); } } polylist_foreach(&sl->Polygon, &it, poly) { pcb_object_operation(opfunc, ctx, PCB_OBJ_POLY, sl, poly, poly); if (!PCB_FLAG_TEST(PCB_FLAG_FLOATER, poly)) { pcb_box_bump_box_noflt(&sc->BoundingBox, &poly->BoundingBox); pcb_box_bump_box_noflt(&sc->bbox_naked, &poly->bbox_naked); } } gfxlist_foreach(&sl->Gfx, &it, gfx) { pcb_object_operation(opfunc, ctx, PCB_OBJ_GFX, sl, gfx, gfx); if (!PCB_FLAG_TEST(PCB_FLAG_FLOATER, gfx)) { pcb_box_bump_box_noflt(&sc->BoundingBox, &gfx->BoundingBox); pcb_box_bump_box_noflt(&sc->bbox_naked, &gfx->bbox_naked); } } } /* execute on globals */ { pcb_pstk_t *ps; gdl_iterator_t it; padstacklist_foreach(&sc->data->padstack, &it, ps) { pcb_object_operation(opfunc, ctx, PCB_OBJ_PSTK, ps, ps, ps); if (!PCB_FLAG_TEST(PCB_FLAG_FLOATER, ps)) { pcb_box_bump_box_noflt(&sc->BoundingBox, &ps->BoundingBox); pcb_box_bump_box_noflt(&sc->bbox_naked, &ps->bbox_naked); } } } rnd_close_box(&sc->BoundingBox); rnd_close_box(&sc->bbox_naked); if (pcb_data_get_top(Data) != NULL) { int doit = 1; if (sc->extobj != NULL) { /* ugly corner case: extobj calls may have added the subc to the rtree already, do not add it twice */ rnd_rtree_it_t it; rnd_box_t *b; if (Data->subc_tree != NULL) { for(b = rnd_rtree_all_first(&it, Data->subc_tree); b != NULL; b = rnd_rtree_all_next(&it)) { if (b == (rnd_box_t *)sc) { doit = 0; break; } } } } if (doit && (Data->subc_tree != NULL)) rnd_rtree_insert(Data->subc_tree, (void *)sc, (rnd_rtree_box_t *)sc); } sc->part_changed_bbox_dirty = 0; /* we've just recalculated the bbox */ pcb_subc_part_changed_inhibit_dec(sc); DrawSubc(sc); switch(batch_undo) { case PCB_SUBCOP_UNDO_SUBC: pcb_undo_unfreeze_add(); pcb_undo_unfreeze_serial(); break; case PCB_SUBCOP_UNDO_BATCH: pcb_undo_unfreeze_serial(); break; case PCB_SUBCOP_UNDO_NORMAL: break; } if (Data != NULL) pcb_data_clip_inhibit_dec(Data, 0); return sc; } /* copies a subcircuit onto the PCB (a.k.a "paste"). Then does a draw. */ void *pcb_subcop_copy(pcb_opctx_t *ctx, pcb_subc_t *src) { pcb_subc_t *sc; sc = pcb_subc_dup_at(PCB, PCB->Data, src, ctx->copy.DeltaX, ctx->copy.DeltaY, ctx->copy.keep_id, rnd_true); pcb_undo_add_obj_to_create(PCB_OBJ_SUBC, sc, sc, sc); if (ctx->copy.from_outside && conf_core.editor.show_solder_side) { uundo_serial_t last; rnd_coord_t w, h; /* move-to-the-other-side is not undoable: it's part of the placement */ pcb_undo_inc_serial(); last = pcb_undo_serial(); pcb_subc_get_origin(sc, &w, &h); pcb_subc_change_side(sc, 2 * h - PCB->hidlib.dwg.Y2); pcb_undo_truncate_from(last); } pcb_text_dyn_bbox_update(sc->data); return sc; } extern pcb_opfunc_t ClipFunctions, MoveFunctions, MoveFunctions_noclip, Rotate90Functions, RotateFunctions, ChgFlagFunctions, ChangeSizeFunctions, ChangeClearSizeFunctions, Change1stSizeFunctions, Change2ndSizeFunctions; /* drag&drop move when not selected */ void *pcb_subcop_move(pcb_opctx_t *ctx, pcb_subc_t *sc) { pcb_board_t *pcb = pcb_data_get_top(sc->data); pcb_data_t *data = (pcb != NULL ? pcb->Data : NULL); pcb_opctx_t clip; /* restore all pins/pads at once, at the old location */ clip.clip.restore = 1; clip.clip.clear = 0; clip.clip.pcb = ctx->move.pcb; pcb_subc_op(data, sc, &ClipFunctions, &clip, PCB_SUBCOP_UNDO_NORMAL); /* do the move without messing with the clipping */ pcb_subc_op(data, sc, &MoveFunctions_noclip, ctx, PCB_SUBCOP_UNDO_NORMAL); /* clear all pins/pads at once, at the new location */ clip.clip.restore = 0; clip.clip.clear = 1; pcb_subc_op(data, sc, &ClipFunctions, &clip, PCB_SUBCOP_UNDO_NORMAL); return sc; } void *pcb_subcop_rotate90(pcb_opctx_t *ctx, pcb_subc_t *sc) { pcb_board_t *pcb = pcb_data_get_top(sc->data); return pcb_subc_op((pcb != NULL ? pcb->Data : NULL), sc, &Rotate90Functions, ctx, PCB_SUBCOP_UNDO_SUBC); } void *pcb_subcop_rotate(pcb_opctx_t *ctx, pcb_subc_t *sc) { pcb_data_t *data; ctx->rotate.pcb = pcb_data_get_top(sc->data); data = (ctx->rotate.pcb != NULL ? ctx->rotate.pcb->Data : NULL); return pcb_subc_op(data, sc, &RotateFunctions, ctx, PCB_SUBCOP_UNDO_SUBC); } void pcb_subc_rotate90(pcb_subc_t *subc, rnd_coord_t cx, rnd_coord_t cy, int steps) { pcb_opctx_t ctx; ctx.rotate.center_x = cx; ctx.rotate.center_y = cy; ctx.rotate.number = steps; pcb_subcop_rotate90(&ctx, subc); } void pcb_subc_rotate(pcb_subc_t *subc, rnd_coord_t cx, rnd_coord_t cy, double cosa, double sina, double angle) { pcb_opctx_t ctx; ctx.rotate.center_x = cx; ctx.rotate.center_y = cy; ctx.rotate.angle = angle; ctx.rotate.cosa = cosa; ctx.rotate.sina = sina; pcb_subcop_rotate(&ctx, subc); } void pcb_subc_move(pcb_subc_t *sc, rnd_coord_t dx, rnd_coord_t dy, rnd_bool more_to_come) { pcb_opctx_t ctx; ctx.move.pcb = NULL; ctx.move.dx = dx; ctx.move.dy = dy; ctx.move.more_to_come = more_to_come; pcb_subcop_move(&ctx, sc); } rnd_bool pcb_selected_subc_change_side(void) { rnd_bool change = rnd_false; if (PCB->pstk_on) { PCB_SUBC_LOOP(PCB->Data); { if (PCB_FLAG_TEST(PCB_FLAG_SELECTED, subc)) { change |= pcb_subc_change_side(subc, 2 * pcb_crosshair.Y - PCB->hidlib.dwg.Y2); } } PCB_END_LOOP; } return change; } static int subc_bind_layer(pcb_layer_t *dl, pcb_layer_t *sl, int dst_is_pcb) { int chg = 0; if (!dst_is_pcb) { /* keep only the layer binding match, unbound other aspects */ sl->meta.bound.real = NULL; sl->arc_tree = sl->line_tree = sl->text_tree = sl->polygon_tree = NULL; chg++; } else { sl->meta.bound.real = dl; chg++; } return chg; } static int subc_relocate_layer_objs(pcb_layer_t *dl, pcb_data_t *src_data, pcb_layer_t *sl, int src_has_real_layer, int dst_is_pcb, int move_obj) { pcb_line_t *line; pcb_text_t *text; pcb_poly_t *poly; pcb_arc_t *arc; pcb_gfx_t *gfx; gdl_iterator_t it; int chg = 0; pcb_data_t *dst_data = dl == NULL ? NULL : dl->parent.data; linelist_foreach(&sl->Line, &it, line) { if (src_has_real_layer) { pcb_poly_restore_to_poly(src_data, PCB_OBJ_LINE, sl, line); rnd_rtree_delete(sl->line_tree, line, (rnd_rtree_box_t *)line); chg++; } PCB_FLAG_CLEAR(PCB_FLAG_WARN | PCB_FLAG_FOUND | PCB_FLAG_SELECTED, line); if ((move_obj) && (dl != NULL)) { pcb_line_unreg(line); pcb_line_reg(dl, line); } if ((dl != NULL) && (dl->line_tree != NULL)) { rnd_rtree_insert(dl->line_tree, line, (rnd_rtree_box_t *)line); chg++; } if (dst_is_pcb && (dl != NULL)) pcb_poly_clear_from_poly(dst_data, PCB_OBJ_LINE, dl, line); } arclist_foreach(&sl->Arc, &it, arc) { if (src_has_real_layer) { pcb_poly_restore_to_poly(src_data, PCB_OBJ_ARC, sl, arc); rnd_rtree_delete(sl->arc_tree, arc, (rnd_rtree_box_t *)arc); chg++; } PCB_FLAG_CLEAR(PCB_FLAG_WARN | PCB_FLAG_FOUND | PCB_FLAG_SELECTED, arc); if ((move_obj) && (dl != NULL)) { pcb_arc_unreg(arc); pcb_arc_reg(dl, arc); } if ((dl != NULL) && (dl->arc_tree != NULL)) { rnd_rtree_insert(dl->arc_tree, arc, (rnd_rtree_box_t *)arc); chg++; } if (dst_is_pcb && (dl != NULL)) pcb_poly_clear_from_poly(dst_data, PCB_OBJ_ARC, dl, arc); } textlist_foreach(&sl->Text, &it, text) { if (src_has_real_layer) { pcb_poly_restore_to_poly(src_data, PCB_OBJ_LINE, sl, text); rnd_rtree_delete(sl->text_tree, text, (rnd_rtree_box_t *)text); chg++; } PCB_FLAG_CLEAR(PCB_FLAG_WARN | PCB_FLAG_FOUND | PCB_FLAG_SELECTED, text); if ((move_obj) && (dl != NULL)) { pcb_text_unreg(text); pcb_text_reg(dl, text); } if ((dl != NULL) && (dl->text_tree != NULL)) { rnd_rtree_insert(dl->text_tree, text, (rnd_rtree_box_t *)text); chg++; } if (dst_is_pcb && (dl != NULL)) pcb_poly_clear_from_poly(dst_data, PCB_OBJ_TEXT, dl, text); } polylist_foreach(&sl->Polygon, &it, poly) { pcb_poly_pprestore(poly); if (src_has_real_layer) { rnd_rtree_delete(sl->polygon_tree, poly, (rnd_rtree_box_t *)poly); chg++; } PCB_FLAG_CLEAR(PCB_FLAG_WARN | PCB_FLAG_FOUND | PCB_FLAG_SELECTED, poly); if ((move_obj) && (dl != NULL)) { pcb_poly_unreg(poly); pcb_poly_reg(dl, poly); } if ((dl != NULL) && (dl->polygon_tree != NULL)) { rnd_rtree_insert(dl->polygon_tree, poly, (rnd_rtree_box_t *)poly); chg++; } if (dst_is_pcb && (dl != NULL)) pcb_poly_ppclear_at(poly, dl); } gfxlist_foreach(&sl->Gfx, &it, gfx) { if (src_has_real_layer) { rnd_rtree_delete(sl->gfx_tree, gfx, (rnd_rtree_box_t *)gfx); chg++; } PCB_FLAG_CLEAR(PCB_FLAG_WARN | PCB_FLAG_FOUND | PCB_FLAG_SELECTED, gfx); if ((move_obj) && (dl != NULL)) { pcb_gfx_unreg(gfx); pcb_gfx_reg(dl, gfx); } if ((dl != NULL) && (dl->gfx_tree != NULL)) { rnd_rtree_insert(dl->gfx_tree, gfx, (rnd_rtree_box_t *)gfx); chg++; } } return chg + subc_bind_layer(dl, sl, dst_is_pcb); } static int subc_bind_globals(pcb_data_t *dst, pcb_subc_t *sc, int dst_is_pcb) { int chg = 0; if ((dst != NULL) && (dst_is_pcb)) { if (dst->padstack_tree == NULL) rnd_rtree_init(dst->padstack_tree = malloc(sizeof(rnd_rtree_t))); sc->data->padstack_tree = dst->padstack_tree; chg++; } else sc->data->padstack_tree = NULL; return chg; } static int subc_relocate_globals(pcb_data_t *dst, pcb_data_t *new_parent, pcb_subc_t *sc, int dst_is_pcb, int move_obj) { pcb_pstk_t *ps; gdl_iterator_t it; int chg = 0; padstacklist_foreach(&sc->data->padstack, &it, ps) { const pcb_pstk_proto_t *proto = pcb_pstk_get_proto(ps); int undoable_psproto = ps->parent.data->parent_type == PCB_PARENT_BOARD; assert(proto != NULL); /* the prototype must be accessible at the source else we can't add it in the dest */ pcb_poly_restore_to_poly(ps->parent.data, PCB_OBJ_PSTK, NULL, ps); if (sc->data->padstack_tree != NULL) rnd_rtree_delete(sc->data->padstack_tree, ps, (rnd_rtree_box_t *)ps); PCB_FLAG_CLEAR(PCB_FLAG_WARN | PCB_FLAG_FOUND | PCB_FLAG_SELECTED, ps); if ((dst != NULL) && (dst->padstack_tree != NULL)) rnd_rtree_insert(dst->padstack_tree, ps, (rnd_rtree_box_t *)ps); if ((move_obj) && (dst != NULL)) { pcb_pstk_unreg(ps); pcb_pstk_reg(dst, ps); } if (dst != NULL) ps->proto = pcb_pstk_proto_insert_dup(ps->parent.data, proto, 1, undoable_psproto); ps->protoi = -1; ps->parent.data = new_parent; if (dst_is_pcb) pcb_poly_clear_from_poly(ps->parent.data, PCB_OBJ_PSTK, NULL, ps); chg++; } return chg + subc_bind_globals(dst, sc, dst_is_pcb); } /* change the parent of all global objects to new_parent */ static void subc_set_parent_globals(pcb_subc_t *sc, pcb_data_t *new_parent) { pcb_data_set_parent_globals(sc->data, new_parent); } void *pcb_subcop_move_buffer(pcb_opctx_t *ctx, pcb_subc_t *sc) { int n; pcb_board_t *dst_top = pcb_data_get_top(ctx->buffer.dst); int dst_is_pcb = ((dst_top != NULL) && (dst_top->Data == ctx->buffer.dst)); pcb_opctx_t clip; EraseSubc(sc); /* move the subc */ if ((ctx->buffer.pcb != NULL) && (ctx->buffer.pcb->Data->subc_tree != NULL)) { if (!dst_is_pcb) { /* restore all pins/pads at once, at the old location */ clip.clip.restore = 1; clip.clip.clear = 0; clip.clip.pcb = ctx->buffer.pcb; pcb_subc_op(ctx->buffer.pcb->Data, sc, &ClipFunctions, &clip, PCB_SUBCOP_UNDO_SUBC); } rnd_rtree_delete(ctx->buffer.pcb->Data->subc_tree, sc, (rnd_rtree_box_t *)sc); } pcb_subc_unreg(sc); pcb_subc_reg(ctx->buffer.dst, sc); if (dst_is_pcb) { if (ctx->buffer.dst->subc_tree == NULL) rnd_rtree_init(ctx->buffer.dst->subc_tree = malloc(sizeof(rnd_rtree_t))); rnd_rtree_insert(ctx->buffer.dst->subc_tree, sc, (rnd_rtree_box_t *)sc); } if (dst_is_pcb) PCB_SET_PARENT(sc, data, ctx->buffer.dst); /* have to set sc parent before relocate globals so that poly clearings will work */ /* move layer local */ for(n = 0; n < sc->data->LayerN; n++) { pcb_layer_t *sl = sc->data->Layer + n; pcb_layer_t *dl; int src_has_real_layer = (sl->meta.bound.real != NULL); if (dst_is_pcb) { dl = pcb_layer_resolve_binding(dst_top, sl); if (dl != NULL) { pcb_layer_link_trees(sl, dl); } else { /* need to create the trees so that move and other ops work */ if (sl->line_tree == NULL) rnd_rtree_init(sl->line_tree = malloc(sizeof(rnd_rtree_t))); if (sl->arc_tree == NULL) rnd_rtree_init(sl->arc_tree = malloc(sizeof(rnd_rtree_t))); if (sl->text_tree == NULL) rnd_rtree_init(sl->text_tree = malloc(sizeof(rnd_rtree_t))); if (sl->polygon_tree == NULL) rnd_rtree_init(sl->polygon_tree = malloc(sizeof(rnd_rtree_t))); if (sl->gfx_tree == NULL) rnd_rtree_init(sl->gfx_tree = malloc(sizeof(rnd_rtree_t))); if (!(sl->meta.bound.type & PCB_LYT_VIRTUAL)) rnd_message(RND_MSG_ERROR, "Couldn't bind subc layer %s on buffer move\n", sl->name == NULL ? "" : sl->name); } } else dl = ctx->buffer.dst->Layer + n; if (dst_is_pcb) { subc_bind_layer(dl, sl, dst_is_pcb); } else { subc_relocate_layer_objs(dl, ctx->buffer.src, sl, src_has_real_layer, dst_is_pcb, 0); /* Happens when moving the subc into buffer, most notably into removelist: - layer objects got registered in dl-> trees - if we don't do the bindings here, subc won't remember its layer objects are registered in which trees - on subc free, layer objects are freed, but rtree del won't run as subc's layer rtree is NULL - layer objects will remain on old dl-> trees (e.g. on removelist trees) - as long as the object is still alive in removelist, it goes on unnoticed; the undo list keeps objects alive; but these are really ghost objects in the rtree because sc free already free'd the layer object - clearing the undo list, {u c}, will flush the remove list and that will expose the bug in form of ghost rtree objects are invalid memory reads now - see TODO:7 @ r38121 (anchor #1) */ if ((sl->line_tree != NULL) || (sl->arc_tree != NULL) || (sl->text_tree != NULL) || (sl->polygon_tree != NULL)) rnd_message(RND_MSG_ERROR, "pcb_subcop_move_buffer(): expected empty layer object trees.\nPlease report this bug.\n"); /* gfx_tree is always bound for some reason (do not free it, it's on the board) */ if (sl->gfx_tree != NULL) sl->gfx_tree = NULL; pcb_layer_link_trees(sl, dl); } } if (dst_is_pcb) { subc_bind_globals(ctx->buffer.dst, sc, dst_is_pcb); /* clear all pins/pads at once, at the new location; this does a relocation of all objects recursively */ clip.clip.restore = 0; clip.clip.clear = 1; clip.clip.pcb = ctx->buffer.pcb; pcb_subc_op(ctx->buffer.dst, sc, &ClipFunctions, &clip, PCB_SUBCOP_UNDO_SUBC); } else { subc_relocate_globals(ctx->buffer.dst, sc->data, sc, dst_is_pcb, 0); /* see anchor #1 above */ sc->data->padstack_tree = ctx->buffer.dst->padstack_tree; } PCB_FLAG_CLEAR(PCB_FLAG_WARN | PCB_FLAG_FOUND | PCB_FLAG_SELECTED, sc); return sc; } void *pcb_subcop_add_to_buffer(pcb_opctx_t *ctx, pcb_subc_t *sc) { pcb_subc_t *nsc; nsc = pcb_subc_dup_at(NULL, ctx->buffer.dst, sc, 0, 0, ctx->buffer.keep_id, rnd_true); if (ctx->buffer.extraflg & PCB_FLAG_SELECTED) { pcb_undo_freeze_serial(); pcb_undo_freeze_add(); pcb_subc_select(NULL, nsc, PCB_CHGFLG_CLEAR, 0); pcb_undo_unfreeze_add(); pcb_undo_unfreeze_serial(); } return nsc; } /* break buffer subc into pieces */ rnd_bool pcb_subc_smash_buffer(pcb_buffer_t *buff) { pcb_subc_t *subc, *next; int n, bn; long warn = 0; for(subc = pcb_subclist_first(&buff->Data->subc); subc != NULL; subc = next) { next = pcb_subclist_next(subc); for(n = 0; n < subc->data->LayerN; n++) { pcb_layer_t *sl = subc->data->Layer + n, *brdl, *dl; if (strcmp(sl->name, SUBC_AUX_NAME) == 0) continue; brdl = pcb_layer_resolve_binding(PCB, sl); dl = NULL; if (brdl != NULL) { for(bn = 0; bn < buff->Data->LayerN; bn++) { if (buff->Data->Layer[bn].meta.bound.real == brdl) { dl = &buff->Data->Layer[bn]; break; } } } if (dl != NULL) subc_relocate_layer_objs(dl, subc->data, sl, 1, 0, 1); else warn++; } subc_relocate_globals(buff->Data, buff->Data, subc, 0, 1); pcb_subc_free(subc); } if (warn) rnd_message(RND_MSG_WARNING, "There are %ld layers that got lost in the smash because they were on unbound subc layers\nThis normally happens if your subcircuits in buffer refer to layers that do not exist on your board.\n", warn); return rnd_true; } int pcb_subc_rebind(pcb_board_t *pcb, pcb_subc_t *sc) { int n, chgly = 0; pcb_board_t *dst_top = pcb_data_get_top(sc->data); int dst_is_pcb = ((dst_top != NULL) && (dst_top->Data == pcb->Data)); if (!dst_is_pcb) return -1; EraseSubc(sc); /* relocate bindings */ for(n = 0; n < sc->data->LayerN; n++) { pcb_layer_t *sl = sc->data->Layer + n; pcb_layer_t *dl = pcb_layer_resolve_binding(dst_top, sl); int src_has_real_layer = (sl->meta.bound.real != NULL); if (dl != NULL) { if (sl->meta.bound.real == dl) continue; /* make sure all trees exist on the dest layer - if these are the first objects there we may need to create them */ if (dl->line_tree == NULL) rnd_rtree_init(dl->line_tree = malloc(sizeof(rnd_rtree_t))); if (dl->arc_tree == NULL) rnd_rtree_init(dl->arc_tree = malloc(sizeof(rnd_rtree_t))); if (dl->text_tree == NULL) rnd_rtree_init(dl->text_tree = malloc(sizeof(rnd_rtree_t))); if (dl->polygon_tree == NULL) rnd_rtree_init(dl->polygon_tree = malloc(sizeof(rnd_rtree_t))); if (dl->gfx_tree == NULL) rnd_rtree_init(dl->gfx_tree = malloc(sizeof(rnd_rtree_t))); } if (subc_relocate_layer_objs(dl, pcb->Data, sl, src_has_real_layer, 1, 0) > 0) chgly++; if (dl != NULL) pcb_layer_link_trees(sl, dl); } return chgly; } void pcb_subc_bind_globals(pcb_board_t *pcb, pcb_subc_t *subc) { if (pcb->Data->padstack_tree == NULL) rnd_rtree_init(pcb->Data->padstack_tree = malloc(sizeof(rnd_rtree_t))); subc->data->padstack_tree = pcb->Data->padstack_tree; TODO("subc: subc-in-subc: bind subc rtree") } /*** undoable unbind ***/ typedef struct { pcb_board_t *pcb; pcb_subc_t *sc; rnd_layer_id_t lid; int slot; /* layer index within the subc */ unsigned int unbind:1; } undo_subc_unbind_t; static int undo_subc_unbind_swap(void *udata) { undo_subc_unbind_t *u = udata; if (u->unbind) subc_relocate_layer_objs(NULL, u->pcb->Data, &u->sc->data->Layer[u->slot], 1, 0, 0); else { pcb_layer_t *ly = pcb_get_layer(u->pcb->Data, u->lid); pcb_layer_link_trees(&u->sc->data->Layer[u->slot], ly); subc_relocate_layer_objs(ly, u->pcb->Data, &u->sc->data->Layer[u->slot], 0, 1, 0); } u->unbind = !u->unbind; return 0; } static void undo_subc_unbind_print(void *udata, char *dst, size_t dst_len) { undo_subc_unbind_t *u = udata; rnd_snprintf(dst, dst_len, "subc unbind layer: %s #%ld:%ld", u->unbind ? "unbind" : "rebind", u->sc->ID, u->slot); } static const uundo_oper_t undo_subc_unbind = { core_subc_cookie, NULL, undo_subc_unbind_swap, undo_subc_unbind_swap, undo_subc_unbind_print }; static void pcb_subc_unbind_(pcb_board_t *pcb, pcb_subc_t *sc, rnd_layer_id_t brdlid, int slot, int undoable) { undo_subc_unbind_t utmp, *u = &utmp; if (undoable) u = pcb_undo_alloc(pcb, &undo_subc_unbind, sizeof(undo_subc_unbind_t)); u->pcb = pcb; u->sc = sc; u->lid = brdlid; u->slot = slot; u->unbind = 1; undo_subc_unbind_swap(u); if (undoable) pcb_undo_inc_serial(); } int pcb_subc_unbind(pcb_board_t *pcb, pcb_subc_t *sc, pcb_layer_t *brdly, int undoable) { int n, res = 0; for(n = 0; n < sc->data->LayerN; n++) { pcb_layer_t *sl = sc->data->Layer + n; if (sl->meta.bound.real == brdly) { pcb_subc_unbind_(pcb, sc, brdly - pcb->Data->Layer, n, undoable); res++; } } if (undoable && (res > 0)) pcb_undo_inc_serial(); return res; } long pcb_subc_unbind_all(pcb_board_t *pcb, pcb_layer_t *brdly, int undoable) { long res = 0; pcb_subc_t *sc; gdl_iterator_t it; subclist_foreach(&pcb->Data->subc, &it, sc) { int r = pcb_subc_unbind(pcb, sc, brdly, undoable); if (r > 0) res += r; } return res; } void *pcb_subcop_change_size(pcb_opctx_t *ctx, pcb_subc_t *sc) { pcb_subc_op(ctx->chgsize.pcb->Data, sc, &ChangeSizeFunctions, ctx, PCB_SUBCOP_UNDO_BATCH); return sc; } void *pcb_subcop_change_clear_size(pcb_opctx_t *ctx, pcb_subc_t *sc) { pcb_subc_op(ctx->chgsize.pcb->Data, sc, &ChangeClearSizeFunctions, ctx, PCB_SUBCOP_UNDO_BATCH); return sc; } void *pcb_subcop_change_1st_size(pcb_opctx_t *ctx, pcb_subc_t *sc) { pcb_subc_op(ctx->chgsize.pcb->Data, sc, &Change1stSizeFunctions, ctx, PCB_SUBCOP_UNDO_BATCH); return sc; } void *pcb_subcop_change_2nd_size(pcb_opctx_t *ctx, pcb_subc_t *sc) { pcb_subc_op(ctx->chgsize.pcb->Data, sc, &Change2ndSizeFunctions, ctx, PCB_SUBCOP_UNDO_BATCH); return sc; } void *pcb_subcop_change_nonetlist(pcb_opctx_t *ctx, pcb_subc_t *sc) { pcb_undo_add_obj_to_flag(sc); if (PCB_FLAG_TEST(PCB_FLAG_LOCK, sc)) return NULL; PCB_FLAG_TOGGLE(PCB_FLAG_NONETLIST, sc); return sc; } void *pcb_subcop_change_name(pcb_opctx_t *ctx, pcb_subc_t *sc) { void *old; if (sc->refdes != NULL) old = rnd_strdup(sc->refdes); /* strdup because the pcb_attribute_put() is going to free the original */ else old = NULL; pcb_attribute_put(&sc->Attributes, "refdes", ctx->chgname.new_name); return old; } void *pcb_subcop_destroy(pcb_opctx_t *ctx, pcb_subc_t *sc) { if (ctx->remove.pcb->Data->subc_tree != NULL) rnd_rtree_delete(ctx->remove.pcb->Data->subc_tree, sc, (rnd_rtree_box_t *)sc); EraseSubc(sc); pcb_subc_free(sc); return NULL; } void *pcb_subcop_remove(pcb_opctx_t *ctx, pcb_subc_t *sc) { if (ctx->remove.pcb->is_footprint) { rnd_message(RND_MSG_WARNING, "Can not remove the subcircuit being edited in the footprint edit mode\n"); return NULL; } EraseSubc(sc); pcb_undo_move_obj_to_remove(PCB_OBJ_SUBC, sc, sc, sc); return NULL; } void *pcb_subc_remove(pcb_subc_t *sc) { void *res; pcb_opctx_t ctx; ctx.remove.pcb = PCB; ctx.remove.destroy_target = NULL; /* PCB_CLEAR_PARENT(subc);*/ res = pcb_subcop_remove(&ctx, sc); pcb_draw(); return res; } void *pcb_subcop_change_flag(pcb_opctx_t *ctx, pcb_subc_t *sc) { static pcb_flag_values_t pcb_subc_flags = 0; if (pcb_subc_flags == 0) pcb_subc_flags = pcb_obj_valid_flags(PCB_OBJ_SUBC); if (ctx->chgflag.flag == PCB_FLAG_FLOATER) { TODO("subc: subc-in-subc: can a whole subc be a floater? - add undo!") PCB_FLAG_CHANGE(ctx->chgflag.how, ctx->chgflag.flag, sc); return sc; } pcb_subc_op(ctx->chgflag.pcb->Data, sc, &ChgFlagFunctions, ctx, PCB_SUBCOP_UNDO_NORMAL); if ((ctx->chgflag.flag & pcb_subc_flags) == ctx->chgflag.flag) { pcb_undo_add_obj_to_flag(sc); PCB_FLAG_CHANGE(ctx->chgflag.how, ctx->chgflag.flag, sc); } return sc; } void pcb_subc_select(pcb_board_t *pcb, pcb_subc_t *sc, pcb_change_flag_t how, int redraw) { pcb_opctx_t ctx; pcb_undo_add_obj_to_flag(sc); ctx.chgflag.pcb = pcb; ctx.chgflag.how = how; ctx.chgflag.flag = PCB_FLAG_SELECTED; pcb_subc_op((pcb == NULL ? NULL : pcb->Data), sc, &ChgFlagFunctions, &ctx, PCB_SUBCOP_UNDO_NORMAL); PCB_FLAG_CHANGE(how, PCB_FLAG_SELECTED, sc); if (redraw) DrawSubc(sc); } /*** undoable mirror ***/ typedef struct { pcb_subc_t *subc; /* it is safe to save the object pointer because it is persistent (through the removed object list) */ rnd_coord_t y_offs; rnd_bool smirror; } undo_subc_mirror_t; static int undo_subc_mirror_swap(void *udata) { undo_subc_mirror_t *g = udata; pcb_data_t *data = g->subc->parent.data; if ((data != NULL) && (data->subc_tree != NULL)) rnd_rtree_delete(data->subc_tree, g->subc, (rnd_rtree_box_t *)g->subc); pcb_undo_freeze_add(); pcb_data_mirror(g->subc->data, g->y_offs, g->smirror ? PCB_TXM_SIDE : PCB_TXM_COORD, g->smirror, 0); pcb_undo_unfreeze_add(); pcb_subc_bbox(g->subc); if ((data != NULL) && (data->subc_tree != NULL)) rnd_rtree_insert(data->subc_tree, g->subc, (rnd_rtree_box_t *)g->subc); return 0; } static void undo_subc_mirror_print(void *udata, char *dst, size_t dst_len) { undo_subc_mirror_t *g = udata; rnd_snprintf(dst, dst_len, "subc mirror y_offs=%$mm smirror=%d", g->y_offs, g->smirror); } static const uundo_oper_t undo_subc_mirror = { core_subc_cookie, NULL, undo_subc_mirror_swap, undo_subc_mirror_swap, undo_subc_mirror_print }; /* mirrors the coordinates of a subcircuit; an additional offset is passed */ void pcb_subc_mirror(pcb_data_t *data, pcb_subc_t *subc, rnd_coord_t y_offs, rnd_bool smirror, rnd_bool undoable) { undo_subc_mirror_t gtmp, *g = >mp; if (undoable) g = pcb_undo_alloc(pcb_data_get_top(subc->parent.data), &undo_subc_mirror, sizeof(undo_subc_mirror_t)); g->subc = subc; g->y_offs = y_offs; g->smirror = smirror; undo_subc_mirror_swap(g); if (undoable) pcb_undo_inc_serial(); } void pcb_subc_scale(pcb_data_t *data, pcb_subc_t *subc, double sx, double sy, double sth, int recurse) { if ((data != NULL) && (data->subc_tree != NULL)) rnd_rtree_delete(data->subc_tree, subc, (rnd_rtree_box_t *)subc); pcb_data_scale(subc->data, sx, sy, sth, recurse); pcb_subc_bbox(subc); if ((data != NULL) && (data->subc_tree != NULL)) rnd_rtree_insert(data->subc_tree, subc, (rnd_rtree_box_t *)subc); } rnd_bool pcb_subc_change_side(pcb_subc_t *subc, rnd_coord_t yoff) { int n; pcb_board_t *pcb; pcb_data_t *data; if (PCB_FLAG_TEST(PCB_FLAG_LOCK, subc)) return rnd_false; pcb_data_clip_inhibit_inc(PCB->Data); assert(subc->parent_type = PCB_PARENT_DATA); data = subc->parent.data; pcb = pcb_data_get_top(data); /* mirror object geometry and stackup */ if ((data != NULL) && (data->subc_tree != NULL)) rnd_rtree_delete(data->subc_tree, subc, (rnd_rtree_box_t *)subc); pcb_undo_freeze_add(); pcb_data_mirror(subc->data, yoff, PCB_TXM_SIDE, 1, 0); pcb_undo_unfreeze_add(); for(n = 0; n < subc->data->LayerN; n++) { pcb_layer_t *ly = subc->data->Layer + n; ly->meta.bound.type = pcb_layer_mirror_type(ly->meta.bound.type); ly->meta.bound.stack_offs = -ly->meta.bound.stack_offs; } pcb_subc_rebind(pcb, subc); pcb_subc_bbox(subc); if ((data != NULL) && (data->subc_tree != NULL)) rnd_rtree_insert(data->subc_tree, subc, (rnd_rtree_box_t *)subc); pcb_undo_add_subc_to_otherside(PCB_OBJ_SUBC, subc, subc, subc, yoff); pcb_data_clip_inhibit_dec(PCB->Data, 0); return rnd_true; } /* in per-side mode, do not show subc mark if it is on the other side */ #define draw_subc_per_layer() \ do { \ int per_side = 0; \ if (cl != NULL) \ per_side = *(int *)cl; \ if (per_side) { \ int on_bottom; \ if (pcb_subc_get_side(subc, &on_bottom) == 0) { \ if ((!!on_bottom) != (!!conf_core.editor.show_solder_side)) \ return rnd_RTREE_DIR_NOT_FOUND_CONT; \ } \ } \ } while(0) #include "conf_core.h" #include "draw.h" rnd_rtree_dir_t pcb_draw_subc_mark(void *cl, void *obj, const rnd_rtree_box_t *box) { pcb_draw_info_t *info = cl; pcb_subc_t *subc = (pcb_subc_t *)obj; rnd_box_t *bb = &subc->BoundingBox; int selected, locked; int freq = conf_core.appearance.subc.dash_freq; const rnd_color_t *nnclr; selected = PCB_FLAG_TEST(PCB_FLAG_SELECTED, subc); locked = PCB_FLAG_TEST(PCB_FLAG_LOCK, subc); draw_subc_per_layer(); if (freq > 32) freq = 32; nnclr = (PCB_FLAG_TEST(PCB_FLAG_NONETLIST, subc)) ? &conf_core.appearance.color.subc_nonetlist : &conf_core.appearance.color.subc; rnd_render->set_color(pcb_draw_out.fgGC, selected ? &conf_core.appearance.color.selected : nnclr); rnd_hid_set_line_width(pcb_draw_out.fgGC, 0); pcb_subc_draw_origin(pcb_draw_out.fgGC, subc, 0, 0); if (locked) { int on_bottom = 0; pcb_subc_get_side(subc, &on_bottom); if (rnd_conf.editor.view.flip_x) on_bottom = !on_bottom; pcb_subc_draw_locked(pcb_draw_out.fgGC, subc, rnd_conf.editor.view.flip_x ? bb->X1 : bb->X2, on_bottom ? bb->Y1 : bb->Y2, on_bottom); } if (freq >= 0) { pcb_draw_dashed_line(info, pcb_draw_out.fgGC, bb->X1, bb->Y1, bb->X2, bb->Y1, freq, rnd_true); pcb_draw_dashed_line(info, pcb_draw_out.fgGC, bb->X1, bb->Y1, bb->X1, bb->Y2, freq, rnd_true); pcb_draw_dashed_line(info, pcb_draw_out.fgGC, bb->X2, bb->Y2, bb->X2, bb->Y1, freq, rnd_true); pcb_draw_dashed_line(info, pcb_draw_out.fgGC, bb->X2, bb->Y2, bb->X1, bb->Y2, freq, rnd_true); } return rnd_RTREE_DIR_FOUND_CONT; } rnd_rtree_dir_t draw_subc_mark_callback(void *cl, void *obj, const rnd_rtree_box_t *b) { pcb_draw_info_t *info = cl; pcb_subc_t *subc = (pcb_subc_t *)obj; pcb_extobj_t *extobj = pcb_extobj_get(subc); if ((extobj != NULL) && (extobj->draw_mark != NULL)) { extobj->draw_mark(info, subc); return rnd_RTREE_DIR_FOUND_CONT; } return pcb_draw_subc_mark(cl, obj, b); } rnd_rtree_dir_t draw_subc_label_callback(void *cl, void *obj, const rnd_rtree_box_t *b) { pcb_draw_info_t *info = cl; pcb_subc_t *subc = (pcb_subc_t *)obj; rnd_box_t *bb = &subc->BoundingBox; rnd_coord_t x0, y0, dx, dy; rnd_font_t *font = &PCB->fontkit.dflt; /* do not display anyting from the other-side subcs */ draw_subc_per_layer(); /* calculate where the label ends up - always top-right, even if board is flipped */ dx = font->max_width/2; dy = font->max_height/2; if (rnd_conf.editor.view.flip_x) { x0 = bb->X2; dx = -dx; } else x0 = bb->X1; if (rnd_conf.editor.view.flip_y) { y0 = bb->Y2; dy = -dy; } else y0 = bb->Y1; /* draw the label from template; if template is not available, draw the refdes */ if ((conf_core.editor.subc_id != NULL) && (*conf_core.editor.subc_id != '\0')) { static gds_t s; s.used = 0; if (pcb_append_dyntext(&s, (pcb_any_obj_t *)subc, conf_core.editor.subc_id) == 0) { char *curr, *next; rnd_coord_t x = x0, y = y0; for(curr = s.array; curr != NULL; curr = next) { int ctrl = 0; next = strchr(curr, '\\'); if (next != NULL) { *next = '\0'; next++; ctrl = 1; } pcb_label_draw(info, x, y, conf_core.appearance.term_label_size/2, 0, 0, curr, 0); if (ctrl) { switch(*next) { case 'n': y += dy; x = x0; break; } next++; } } } else pcb_label_draw(info, x0, y0, conf_core.appearance.term_label_size/2.0, 0, 0, "", 0); } else if (subc->refdes != NULL) pcb_label_draw(info, x0, y0, conf_core.appearance.term_label_size/2.0, 0, 0, subc->refdes, 0); return rnd_RTREE_DIR_FOUND_CONT; } static int subc_prev_layer_vis(pcb_layer_t *layer) { pcb_layer_t *real = pcb_layer_get_real(layer); if (real == NULL) return 1; return real->meta.real.vis; } static void subc_prev_draw_pstk(const pcb_subc_t *sc, const rnd_box_t *drawn_area, pcb_draw_info_t *info) { rnd_rtree_it_t it; pcb_any_obj_t *o; if (sc->data->padstack_tree == NULL) return; for(o = rnd_rtree_first(&it, sc->data->padstack_tree, (rnd_rtree_box_t *)drawn_area); o != NULL; o = rnd_rtree_next(&it)) { if (pcb_obj_is_under(o, sc->data)) { pcb_pstk_draw_callback(info, (void *)o, (rnd_rtree_box_t *)o); if (PCB->hole_on) pcb_pstk_draw_hole_callback(info, (void *)o, (rnd_rtree_box_t *)o); } } } static void subc_prev_draw_by_lyt(const pcb_subc_t *sc, const rnd_box_t *drawn_area, pcb_draw_info_t *info, pcb_layer_type_t lyt_mask, pcb_layer_combining_t comb) { int n; /* draw layer objects on matching layers */ for(n = 0; n < sc->data->LayerN; n++) { pcb_layer_t *layer = &sc->data->Layer[n]; if (((layer->meta.bound.type & lyt_mask) == lyt_mask) && subc_prev_layer_vis(layer)) pcb_draw_layer_under(PCB, layer, drawn_area, sc->data, info->xform_caller); } /* draw padstack shapes on matching layers */ for(n = 0; n < sc->parent.data->LayerN; n++) { pcb_layer_t *layer = &sc->parent.data->Layer[n]; pcb_layer_type_t lyt = pcb_layer_flags_(layer); if (((lyt & lyt_mask) == lyt_mask) && subc_prev_layer_vis(layer)) { info->objcb.pstk.shape_mask = lyt; info->objcb.pstk.comb = comb; info->objcb.pstk.layer1 = layer; subc_prev_draw_pstk(sc, drawn_area, info); } } } void pcb_subc_draw_preview(const pcb_subc_t *sc, const rnd_box_t *drawn_area) { int n; pcb_draw_info_t info = {0}; rnd_rtree_it_t it; pcb_any_obj_t *o; rnd_xform_t xf = {0}; pcb_draw_setup_default_gui_xform(&xf); /* draw padstacks */ info.pcb = PCB; info.drawn_area = drawn_area; info.xform_caller = info.xform_exporter = info.xform = &xf; info.layer = NULL; info.objcb.pstk.gid = -1; info.objcb.pstk.is_current = 1; info.objcb.pstk.comb = 0; info.objcb.pstk.holetype = PCB_PHOLE_UNPLATED | PCB_PHOLE_PLATED; subc_prev_draw_by_lyt(sc, drawn_area, &info, PCB_LYT_MASK | PCB_LYT_TOP, PCB_LYC_AUTO | PCB_LYC_SUB); subc_prev_draw_by_lyt(sc, drawn_area, &info, PCB_LYT_COPPER, 0); subc_prev_draw_by_lyt(sc, drawn_area, &info, PCB_LYT_PASTE | PCB_LYT_TOP, PCB_LYC_AUTO); /* draw silk and mech and doc layers, above padstacks */ for(n = 0; n < sc->data->LayerN; n++) { pcb_layer_t *layer = &sc->data->Layer[n]; if (((layer->meta.bound.type & (PCB_LYT_SILK | PCB_LYT_BOUNDARY | PCB_LYT_MECH | PCB_LYT_DOC)) && subc_prev_layer_vis(layer))) pcb_draw_layer_under(PCB, layer, drawn_area, sc->data, &xf); } /* padstack mark goes on top */ if (sc->data->padstack_tree != NULL) for(o = rnd_rtree_first(&it, sc->data->padstack_tree, (rnd_rtree_box_t *)drawn_area); o != NULL; o = rnd_rtree_next(&it)) { if (pcb_obj_is_under(o, sc->data)) { pcb_pstk_draw_mark_callback(&info, (void *)o, (rnd_rtree_box_t *)o); pcb_pstk_draw_label_callback(&info, (void *)o, (rnd_rtree_box_t *)o); } } } pcb_subc_t *pcb_subc_by_refdes(pcb_data_t *base, const char *name) { TODO("subc: subc-in-subc hierarchy") PCB_SUBC_LOOP(base); { if ((subc->refdes != NULL) && (RND_NSTRCMP(subc->refdes, name) == 0)) return subc; } PCB_END_LOOP; return NULL; } pcb_subc_t *pcb_subc_by_id(pcb_data_t *base, long int ID) { /* We can not have an rtree based search here: we are often called in the middle of an operation, after the subc got already removed from the rtree. It happens in e.g. undoable padstack operations where the padstack tries to look up its parent subc by ID, while the subc is being rotated. The solution will be the ID hash. */ TODO("subc: subc-in-subc hierarchy") PCB_SUBC_LOOP(base); { if (subc->ID == ID) return subc; } PCB_END_LOOP; return NULL; } rnd_bool pcb_subc_is_empty(pcb_subc_t *subc) { return pcb_data_is_empty(subc->data); } pcb_layer_t *pcb_subc_get_layer(pcb_subc_t *sc, pcb_layer_type_t lyt, pcb_layer_combining_t comb, rnd_bool_t alloc, const char *name, rnd_bool req_name_match) { int n; /* look for an existing layer with matching lyt and comb first */ for(n = 0; n < sc->data->LayerN; n++) { if (sc->data->Layer[n].meta.bound.type != lyt) continue; if ((comb != -1) && (sc->data->Layer[n].comb != comb)) continue; if (req_name_match && (strcmp(sc->data->Layer[n].name, name) != 0)) continue; return &sc->data->Layer[n]; } if (!alloc) return NULL; if (sc->data->LayerN == PCB_MAX_LAYER) return NULL; n = sc->data->LayerN++; if (name == NULL) name = ""; if (comb == -1) { /* "any" means default, for the given layer type */ if (lyt & PCB_LYT_MASK) comb = PCB_LYC_SUB; else comb = 0; /* positive, manual */ } memset(&sc->data->Layer[n], 0, sizeof(sc->data->Layer[n])); sc->data->Layer[n].name = rnd_strdup(name); sc->data->Layer[n].comb = comb; sc->data->Layer[n].is_bound = 1; sc->data->Layer[n].meta.bound.type = lyt; sc->data->Layer[n].parent.data = sc->data; sc->data->Layer[n].parent_type = PCB_PARENT_DATA; sc->data->Layer[n].type = PCB_OBJ_LAYER; return &sc->data->Layer[n]; } RND_INLINE void pcb_subc_part_changed__(pcb_subc_t *sc, int force) { /* can't do this incrementally: if a boundary object is smaller than before it has to make the subc bbox smaller too */ if (force || sc->part_changed_bbox_dirty) { pcb_subc_bbox(sc); sc->part_changed_bbox_dirty = 0; } } void pcb_subc_part_changed_(pcb_any_obj_t *obj) { pcb_subc_t *sc = pcb_obj_parent_subc(obj); if (sc == NULL) return; if (sc->part_changed_inhibit) sc->part_changed_bbox_dirty = 1; else pcb_subc_part_changed__(sc, 1); } void pcb_subc_part_changed_inhibit_inc(pcb_subc_t *sc) { sc->part_changed_inhibit++; } void pcb_subc_part_changed_inhibit_dec(pcb_subc_t *sc) { if (sc->part_changed_inhibit > 0) { sc->part_changed_inhibit--; if (sc->part_changed_inhibit == 0) pcb_subc_part_changed__(sc, 0); } else rnd_message(RND_MSG_ERROR, "Internal error: pcb_subc_part_changed_inhibit_dec(): underflow; please reprot this bug\n"); } const char *pcb_subc_name(pcb_subc_t *subc, const char *local_name) { const char *val; if (local_name != NULL) { val = pcb_attribute_get(&subc->Attributes, local_name); if (val != NULL) return val; } val = pcb_attribute_get(&subc->Attributes, "visible_footprint"); if (val != NULL) return val; val = pcb_attribute_get(&subc->Attributes, "footprint"); return val; } /*** subc replace: in-place replacement of a subcircuit with another, trying to keep as much local changes as possible ***/ typedef struct pcb_sucbrepl_floater_s pcb_sucbrepl_floater_t; struct pcb_sucbrepl_floater_s { /* digest of a floater for matching up */ /* desired state */ rnd_coord_t x, y; double rot; /* admiin */ pcb_sucbrepl_floater_t *next; /* matching */ pcb_objtype_t type; rnd_layer_id_t lid; char str[1]; /* dynamic length */ }; typedef struct { pcb_board_t *pcb; htsp_t thermal; /* value is (char *) thermal shapes; it's as long as many layers pcb currently has */ pcb_sucbrepl_floater_t *flt; } pcb_subc_replace_t; static void pcb_subcrepl_map_floater(pcb_subc_replace_t *repl, const pcb_subc_t *src, pcb_any_obj_t *o) { pcb_sucbrepl_floater_t *f; pcb_text_t *txt = (pcb_text_t *)o; if ((txt->type == PCB_OBJ_TEXT) && (txt->TextString != NULL) && (*txt->TextString != '\0')) { long len = strlen(txt->TextString); f = malloc(sizeof(pcb_sucbrepl_floater_t) + len + 1); strcpy(f->str, txt->TextString); } else { f = malloc(sizeof(pcb_sucbrepl_floater_t)); f->str[0] = '\0'; } f->type = o->type; if (PCB_OBJ_CLASS_LAYER & o->type) f->lid = pcb_layer2id(repl->pcb->Data, pcb_layer_get_real(o->parent.layer)); else f->lid = -1; switch(o->type) { case PCB_OBJ_ARC: f->x = ((pcb_arc_t *)o)->X; f->y = ((pcb_arc_t *)o)->Y; break; case PCB_OBJ_LINE: f->x = ((pcb_line_t *)o)->Point1.X; f->y = ((pcb_line_t *)o)->Point1.Y; break; case PCB_OBJ_TEXT: f->x = ((pcb_text_t *)o)->X; f->y = ((pcb_text_t *)o)->Y; f->rot = ((pcb_text_t *)o)->rot; break; case PCB_OBJ_PSTK: f->x = ((pcb_pstk_t *)o)->x; f->y = ((pcb_pstk_t *)o)->y; f->rot = ((pcb_pstk_t *)o)->rot; break; case PCB_OBJ_GFX: f->x = ((pcb_gfx_t *)o)->cx; f->y = ((pcb_gfx_t *)o)->cy; f->rot = ((pcb_gfx_t *)o)->rot; break; case PCB_OBJ_SUBC: /* may matter in subc-in-subc, for subc terminals */ case PCB_OBJ_POLY: /* no idea which point to use */ case PCB_OBJ_RAT: case PCB_OBJ_NET: case PCB_OBJ_NET_TERM: case PCB_OBJ_LAYER: case PCB_OBJ_LAYERGRP: case PCB_OBJ_VOID: break; } f->next = repl->flt; repl->flt = f; } static void pcb_subcrepl_apply_floater(pcb_subc_replace_t *repl, const pcb_subc_t *src, pcb_any_obj_t *o) { pcb_sucbrepl_floater_t *n; rnd_layer_id_t lid; for(n = repl->flt; n != NULL; n = n->next) { if (n->type != o->type) continue; if (PCB_OBJ_CLASS_LAYER & o->type) lid = pcb_layer2id(repl->pcb->Data, pcb_layer_get_real(o->parent.layer)); else lid = -1; if (n->lid != lid) continue; if (o->type == PCB_OBJ_TEXT) { const char *s = ((pcb_text_t *)o)->TextString; if (s == NULL) s = ""; if (strcmp(n->str, s) != 0) continue; } /* match! */ switch(o->type) { case PCB_OBJ_ARC: pcb_move_obj(o->type, o->parent.any, o, o, n->x - ((pcb_arc_t *)o)->X, n->y - ((pcb_arc_t *)o)->Y); break; case PCB_OBJ_LINE: pcb_move_obj(o->type, o->parent.any, o, o, n->x - ((pcb_line_t *)o)->Point1.X, n->y - ((pcb_line_t *)o)->Point1.Y); break; case PCB_OBJ_TEXT: pcb_move_obj(o->type, o->parent.any, o, o, n->x - ((pcb_text_t *)o)->X, n->y - ((pcb_text_t *)o)->Y); if (n->rot != ((pcb_text_t *)o)->rot) { double ang = n->rot - ((pcb_text_t *)o)->rot; pcb_text_rotate((pcb_text_t *)o, ((pcb_text_t *)o)->X, ((pcb_text_t *)o)->Y, cos(ang), sin(ang), ang); } break; case PCB_OBJ_PSTK: pcb_move_obj(o->type, o->parent.any, o, o, n->x - ((pcb_pstk_t *)o)->x, n->y - ((pcb_pstk_t *)o)->y); if (n->rot != ((pcb_pstk_t *)o)->rot) { double ang = n->rot - ((pcb_text_t *)o)->rot; pcb_pstk_rotate((pcb_pstk_t *)o, ((pcb_pstk_t *)o)->x, ((pcb_pstk_t *)o)->y, cos(ang), sin(ang), ang); } break; case PCB_OBJ_GFX: pcb_gfx_chg_geo((pcb_gfx_t *)o, n->x, n->y, ((pcb_gfx_t *)o)->sx, ((pcb_gfx_t *)o)->sy, n->rot, 0); break; case PCB_OBJ_SUBC: /* may matter in subc-in-subc, for subc terminals */ case PCB_OBJ_POLY: /* no idea which point to use */ case PCB_OBJ_RAT: case PCB_OBJ_NET: case PCB_OBJ_NET_TERM: case PCB_OBJ_LAYER: case PCB_OBJ_LAYERGRP: case PCB_OBJ_VOID: break; } n->type = PCB_OBJ_VOID; /* use each input floater only once */ } } static void pcb_subcrepl_free_floaters(pcb_subc_replace_t *repl) { pcb_sucbrepl_floater_t *n, *next; for(n = repl->flt; n != NULL; n = next) { next = n->next; free(n); } } /* map thermals and floaters of src to repl */ static void pcb_subcrepl_map_thermals_floaters(pcb_subc_replace_t *repl, const pcb_subc_t *src) { pcb_any_obj_t *o; pcb_data_it_t it; char *s; htsp_init(&repl->thermal, strhash, strkeyeq); for(o = pcb_data_first(&it, src->data, PCB_OBJ_CLASS_LAYER); o != NULL; o = pcb_data_next(&it)) { rnd_layer_id_t lid; if (PCB_FLAG_TEST(PCB_FLAG_FLOATER, o)) pcb_subcrepl_map_floater(repl, src, o); if ((o->term == NULL) || (o->thermal == 0)) continue; s = htsp_get(&repl->thermal, o->term); if (s != NULL) continue; /* save the "first" only */ lid = pcb_layer2id(repl->pcb->Data, pcb_layer_get_real(o->parent.layer)); if (lid == -1) continue; s = calloc(repl->pcb->Data->LayerN, 1); s[lid] = o->thermal; htsp_set(&repl->thermal, rnd_strdup(o->term), s); } for(o = pcb_data_first(&it, src->data, PCB_OBJ_PSTK); o != NULL; o = pcb_data_next(&it)) { pcb_pstk_t *ps = (pcb_pstk_t *)o; if (PCB_FLAG_TEST(PCB_FLAG_FLOATER, o)) pcb_subcrepl_map_floater(repl, src, o); if ((o->term == NULL) || (ps->thermals.used == 0)) continue; s = htsp_get(&repl->thermal, o->term); if (s != NULL) continue; /* save the "first" only */ s = calloc(repl->pcb->Data->LayerN, 1); memcpy(s, ps->thermals.shape, ps->thermals.used); htsp_set(&repl->thermal, rnd_strdup(o->term), s); } } /* Set thermals on the new subc terminals from repl; return 1 if there was any thermal set (and thus polygons need to be updated); also move/rotate floaters trying to match the original placement */ static int pcb_subcrepl_apply_thermals_floaters(pcb_subc_replace_t *repl, pcb_subc_t *placed) { pcb_any_obj_t *o; pcb_data_it_t it; htsp_entry_t *e; int has_thermal = 0; for(o = pcb_data_first(&it, placed->data, PCB_OBJ_CLASS_LAYER); o != NULL; o = pcb_data_next(&it)) { rnd_layer_id_t lid; char *s; if (PCB_FLAG_TEST(PCB_FLAG_FLOATER, o)) pcb_subcrepl_apply_floater(repl, placed, o); if ((o->term == NULL) || (o->thermal == 0)) continue; e = htsp_popentry(&repl->thermal, o->term); if (e == NULL) continue; /* save the "first" only */ lid = pcb_layer2id(repl->pcb->Data, pcb_layer_get_real(o->parent.layer)); if (lid == -1) continue; s = e->value; o->thermal = s[lid]; free(e->key); free(e->value); has_thermal = 1; } for(o = pcb_data_first(&it, placed->data, PCB_OBJ_PSTK); o != NULL; o = pcb_data_next(&it)) { pcb_pstk_t *ps = (pcb_pstk_t *)o; if (PCB_FLAG_TEST(PCB_FLAG_FLOATER, o)) pcb_subcrepl_apply_floater(repl, placed, o); if ((o->term == NULL) || (ps->thermals.used == 0)) continue; e = htsp_popentry(&repl->thermal, o->term); if (e == NULL) continue; /* save the "first" only */ ps->thermals.used = repl->pcb->Data->LayerN; ps->thermals.shape = e->value; free(e->key); has_thermal = 1; } genht_uninit_deep(htsp, &repl->thermal, { free(htent->key); free(htent->value); }); pcb_subcrepl_free_floaters(repl); return has_thermal; } pcb_subc_t *pcb_subc_replace(pcb_board_t *pcb, pcb_subc_t *dst, pcb_subc_t *src, rnd_bool add_undo, rnd_bool dumb) { pcb_data_t *data = dst->parent.data; pcb_subc_t *placed; rnd_coord_t ox, oy, osx, osy; double rot = 0; long int target_id; int dst_on_bottom = 0, src_on_bottom = 0; static const pcb_flag_values_t fmask = PCB_FLAG_SELECTED | \ PCB_FLAG_ONSOLDER | PCB_FLAG_AUTO | PCB_FLAG_NONETLIST; pcb_flag_values_t flags; pcb_subc_replace_t repl = {0}; assert(dst->parent_type == PCB_PARENT_DATA); assert(src != NULL); if (dumb) { ox = pcb_crosshair.X; oy = pcb_crosshair.Y; } else { if (pcb_subc_get_origin(dst, &ox, &oy) != 0) { ox = (dst->BoundingBox.X1 + dst->BoundingBox.X2) / 2; oy = (dst->BoundingBox.Y1 + dst->BoundingBox.Y2) / 2; } pcb_subc_get_rotation(dst, &rot); pcb_subc_get_side(dst, &dst_on_bottom); } repl.pcb = pcb; pcb_subcrepl_map_thermals_floaters(&repl, dst); if (pcb_subc_get_origin(src, &osx, &osy) != 0) { osx = (dst->BoundingBox.X1 + dst->BoundingBox.X2) / 2; osy = (dst->BoundingBox.Y1 + dst->BoundingBox.Y2) / 2; } placed = pcb_subc_dup_at(pcb, data, src, ox - osx, oy - osy, 0, rnd_true); pcb_subc_get_side(placed, &src_on_bottom); { /* copy attributes */ int n; pcb_attribute_list_t *adst = &placed->Attributes, *asrc = &dst->Attributes; for (n = 0; n < asrc->Number; n++) if (strcmp(asrc->List[n].name, "footprint") != 0) pcb_attribute_put(adst, asrc->List[n].name, asrc->List[n].value); } flags = dst->Flags.f & fmask; target_id = dst->ID; pcb_subc_remove(dst); pcb_obj_id_del(data, placed); /* NOTE: Can not keep ID because that would fool the undo system: double-replace a subc and there will be two instances with the same ID on the undo list and searching by ID will get fooled! TODO27 */ /* placed->ID = target_id;*/ (void)target_id; pcb_obj_id_reg(data, placed); placed->Flags.f &= ~fmask; placed->Flags.f |= flags; if (rot != 0) { if (dst_on_bottom != src_on_bottom) rot = -rot; pcb_subc_rotate(placed, ox, oy, cos(rot / RND_RAD_TO_DEG), sin(rot / RND_RAD_TO_DEG), rot); } if (dst_on_bottom != src_on_bottom) pcb_subc_change_side(placed, 2 * oy - PCB->hidlib.dwg.Y2); pcb_undo_freeze_add(); if (pcb_subcrepl_apply_thermals_floaters(&repl, placed)) { pcb_opctx_t clip; clip.clip.pcb = pcb; clip.clip.restore = 1; clip.clip.clear = 0; pcb_subc_op(pcb->Data, placed, &ClipFunctions, &clip, PCB_SUBCOP_UNDO_NORMAL); clip.clip.restore = 0; clip.clip.clear = 1; pcb_subc_op(pcb->Data, placed, &ClipFunctions, &clip, PCB_SUBCOP_UNDO_NORMAL); } pcb_subc_select(pcb, placed, (flags & PCB_FLAG_SELECTED) ? PCB_CHGFLG_SET : PCB_CHGFLG_CLEAR, 0); pcb_undo_unfreeze_add(); if (add_undo) pcb_undo_add_obj_to_create(PCB_OBJ_SUBC, placed, placed, placed); return placed; }