import bpy from bpy.types import Operator import bgl from gpu_extras.presets import draw_circle_2d from gpu_extras.batch import batch_for_shader import gpu from time import time from mathutils import Vector, Matrix, Euler from mathutils.kdtree import KDTree from mathutils.geometry import intersect_line_plane, intersect_line_sphere_2d, intersect_line_line from bpy_extras.view3d_utils import region_2d_to_location_3d, region_2d_to_vector_3d, \ location_3d_to_region_2d, region_2d_to_origin_3d, region_2d_to_location_3d from time import time from math import pi, cos, sin def get_gp_mat(gp, name, set_active=False): mat = bpy.data.materials.get(name) if not mat: mat = bpy.data.materials.new(name) bpy.data.materials.create_gpencil_data(mat) if mat not in gp.data.materials[:]: gp.data.materials.append(mat) mat_index = gp.data.materials[:].index(mat) if set_active: gp.active_material_index = mat_index return mat def get_gp_frame(gp_layer, frame=None): if frame is None: frame = bpy.context.scene.frame_current gp_frame = next((f for f in gp_layer.frames if f.frame_number==frame), None) if not gp_frame: gp_frame = gp_layer.frames.new(frame) return gp_frame def get_gp_layer(gp, name=None): if not name: return gp.data.layers.active layer = gp.data.layers.get(name) if not layer: layer = gp.data.layers.new(name) gp.data.layers.active = layer return layer def co_2d_to_3d(co, depth=0.1): area = bpy.context.area region = bpy.context.region rv3d = area.spaces.active.region_3d view_mat = rv3d.view_matrix.inverted() org = view_mat.to_translation() depth_3d = view_mat @ Vector((0, 0, -depth)) #org = region_2d_to_origin_3d(region, rv3d, (region.width/2.0, region.height/2.0)) return region_2d_to_location_3d(region, rv3d, co, depth_3d) #vec = (region_2d_to_origin_3d(region, rv3d, co) - org).normalized() return org + vec return org + region_2d_to_vector_3d(region, rv3d , co) def get_cuts_data(strokes, mouse, radius): gp = bpy.context.object area = bpy.context.area region = bpy.context.region rv3d = area.spaces.active.region_3d view_mat = rv3d.view_matrix.inverted() org = view_mat.to_translation() mat = gp.matrix_world cuts_data = [] for s in strokes: is_polyline = 2<=len(s.points)<=5 if not is_polyline and not s.select: continue print('Cut Stroke', s) for i, p in enumerate(s.points): if not p.select and not is_polyline: continue # Test if the next or previous is unselected edges = [] if i > 0: prev_p = s.points[i-1] if not prev_p.select and not is_polyline: edges.append((i-1, i)) if i < len(s.points)-1: next_p = s.points[i+1] if not next_p.select or is_polyline: edges.append((i, i+1)) for p1_index, p2_index in edges: p1 = s.points[p1_index] p2 = s.points[p2_index] length_3d = (p2.co-p1.co).length p1_3d = mat @ p1.co p2_3d = mat @ p2.co p1_2d = location_3d_to_region_2d(region, rv3d, p1_3d) p2_2d = location_3d_to_region_2d(region, rv3d, p2_3d) if p1_2d is None or p2_2d is None: continue length_2d = (p2_2d-p1_2d).length if length_2d <= 4: continue intersects = intersect_line_sphere_2d(p1_2d, p2_2d, mouse, radius+2) intersects = [i for i in intersects if i is not None] if not intersects: continue close_points = [(p1_2d-i).length < 1 or (p2_2d-i).length < 1 for i in intersects] if any(close_points): #print('close_points', close_points) continue print('intersects', intersects) line_intersects = [] for i_2d in intersects: #factor = ((i_2d-p1_2d).length) / length_2d #factor_3d = factor_2d * length_3d #vec = region_2d_to_vector_3d(region, rv3d, i_2d) #p3_3d = region_2d_to_location_3d(region, rv3d, i_2d, org) #p4_3d = region_2d_to_origin_3d(region, rv3d, i_2d) p3_3d = co_2d_to_3d(i_2d, 0.1) p4_3d = co_2d_to_3d(i_2d, 1000) #bpy.context.scene.cursor.location = p4_3d line_intersect = intersect_line_line(p1_3d, p2_3d, p3_3d, p4_3d) if not line_intersect: continue i1_3d, _ = line_intersect line_intersects += [i1_3d] #context.scene.cursor.location = i1_3d print('line_intersects', line_intersects) if line_intersects: line_intersects.sort(key=lambda x : (x-p1_3d).length) #cut_data[-1].sort(key=lambda x : (x-p1_3d).length) cut_data = [p1_index, p2_index, s, line_intersects] cuts_data.append(cut_data) return cuts_data def circle(x, y, radius, segments): coords = [] m = (1.0 / (segments - 1)) * (pi * 2) for p in range(segments): p1 = x + cos(m * p) * radius p2 = y + sin(m * p) * radius coords.append((p1, p2)) return coords class GPTB_OT_eraser(Operator): """Draw a line with the mouse""" bl_idname = "gp.eraser" bl_label = "Eraser Brush" bl_options = {'REGISTER', 'UNDO'} def draw_callback_px(self): bgl.glEnable(bgl.GL_BLEND) #bgl.glBlendFunc(bgl.GL_CONSTANT_ALPHA, bgl.GL_ONE_MINUS_CONSTANT_ALPHA) #bgl.glBlendColor(1.0, 1.0, 1.0, 0.1) area = bpy.context.area #region = bpy.context.region #rv3d = area.spaces.active.region_3d bg_color = area.spaces.active.shading.background_color #print(bg_color) shader = gpu.shader.from_builtin('2D_UNIFORM_COLOR') shader.bind() shader.uniform_float("color", (1, 1, 1, 1)) for mouse, radius in self.mouse_path: circle_co = circle(*mouse, radius, 24) batch = batch_for_shader(shader, 'TRI_FAN', {"pos": circle_co}) batch.draw(shader) draw_circle_2d(self.mouse, (0.75, 0.25, 0.35, 1.0), self.radius, 24) bgl.glDisable(bgl.GL_BLEND) ''' def draw_holdout(self, context, event): gp = context.object mat_inv = gp.matrix_world.inverted() mouse_3d = co_2d_to_3d(self.mouse) radius_3d = co_2d_to_3d(self.mouse + Vector((self.radius, 0))) search_radius = (radius_3d-mouse_3d).length #print('search_radius', search_radius) #print('radius', self.radius) #bpy.context.scene.cursor.location = mouse_3d for gp_frame, hld_stroke in self.hld_strokes: #print('Add Point') hld_stroke.points.add(count=1) p = hld_stroke.points[-1] p.co = mat_inv @ mouse_3d p.pressure = search_radius * 2000 #context.scene.cursor.location = mouse_3d ''' def get_radius(self, context, event): pressure = event.pressure or 1 return context.scene.gptoolprops.eraser_radius * pressure def erase(self, context, event): gp = context.object mat_inv = gp.matrix_world.inverted() new_points = [] #print(self.cuts_data) # for f in self.gp_frames: # for s in [s for s in f.strokes if s.material_index==self.hld_index]: # f.strokes.remove(s) #gp.data.materials.pop(index=self.hld_index) #bpy.data.materials.remove(self.hld_mat) bpy.ops.object.mode_set(mode='EDIT_GPENCIL') context.scene.tool_settings.gpencil_selectmode_edit = 'POINT' #context.scene.tool_settings.gpencil_selectmode_edit = 'POINT' #bpy.ops.gpencil.select_circle(x=x, y=y, radius=radius, wait_for_input=False) #for cut_data in self.cuts_data: # print(cut_data, len(cut_data)) t0 = time() print() print('Number of cuts', len(self.mouse_path)) for mouse, radius in self.mouse_path: t1 = time() print() x, y = mouse bpy.ops.gpencil.select_all(action='DESELECT') bpy.ops.gpencil.select_circle(x=x, y=y, radius=radius, wait_for_input=False) strokes = [s for f in self.gp_frames for s in f.strokes] #print('select_circle', time()-t1) t2 = time() cut_data = get_cuts_data(strokes, mouse, radius) #print('get_cuts_data', time()-t2) #print([s for s in strokes if s.select]) print('cut_data', cut_data) t3 = time() for p1_index, p2_index, stroke, intersects in cut_data[::-1]: bpy.ops.gpencil.select_all(action='DESELECT') #print('p1_index', p1_index) #print('p2_index', p2_index) p1 = stroke.points[p1_index] p2 = stroke.points[p2_index] p1.select = True p2.select = True number_cuts = len(intersects) bpy.ops.gpencil.stroke_subdivide(number_cuts=number_cuts, only_selected=True) new_p1 = stroke.points[p1_index+1] new_p1.co = mat_inv@intersects[0] new_points += [(stroke, p1_index+1)] #print('number_cuts', number_cuts) if number_cuts == 2: new_p2 = stroke.points[p1_index+2] new_p2.co = mat_inv@( (intersects[0] + intersects[1])/2 ) #new_points += [new_p2] new_p3 = stroke.points[p1_index+3] new_p3.co = mat_inv@intersects[1] new_points += [(stroke, p1_index+3)] #print('subdivide', time() - t3) bpy.ops.gpencil.select_all(action='DESELECT') bpy.ops.gpencil.select_circle(x=x, y=y, radius=radius, wait_for_input=False) ''' selected_strokes = [s for f in self.gp_frames for s in f.strokes if s.select] tip_points = [p for s in selected_strokes for i, p in enumerate(s.points) if p.select and (i==0 or i == len(s.points)-1)] bpy.ops.gpencil.select_less() for p in tip_points: p.select = True for stroke, index in new_points: stroke.points[index].select = False ''' t4 = time() selected_strokes = [s for f in self.gp_frames for s in f.strokes if s.select] if selected_strokes: bpy.ops.gpencil.delete(type='POINTS') print('remove points', time()- t4) #print('Total one cut', time()-t1) #print('Total all cuts', time()-t0) #bpy.ops.gpencil.select_less() #for stroke, index in new_points: # stroke.points[index].select = False #bpy.ops.object.mode_set(mode='OBJECT') bpy.ops.object.mode_set(mode='PAINT_GPENCIL') #selected_strokes = [s for s in self.gp_frame.strokes if s.select] #bpy.ops.object.mode_set(mode='PAINT_GPENCIL') def modal(self, context, event): self.mouse = Vector((event.mouse_region_x, event.mouse_region_y)) self.radius = self.get_radius(context, event) context.area.tag_redraw() if event.type == 'LEFTMOUSE': #self.mouse = mouse #self.mouse_path.append((self.mouse, self.radius)) self.erase(context, event) bpy.types.SpaceView3D.draw_handler_remove(self._handle, 'WINDOW') return {'FINISHED'} if (self.mouse-self.mouse_prev).length < max(self.radius/1.33, 2): return {'RUNNING_MODAL'} self.mouse_prev = self.mouse if event.type in {'MOUSEMOVE', 'INBETWEEN_MOUSEMOVE'}: #self.draw_holdout(context, event) self.mouse_path.append((self.mouse, self.radius)) #self.update_cuts_data(context, event) #self.erase(context, event) return {'RUNNING_MODAL'} elif event.type in {'RIGHTMOUSE', 'ESC'}: bpy.types.SpaceView3D.draw_handler_remove(self._handle, 'WINDOW') return {'CANCELLED'} return {'RUNNING_MODAL'} def invoke(self, context, event): gp = context.object matrix = gp.matrix_world self.radius = self.get_radius(context, event) self.mouse_prev = self.mouse = Vector((event.mouse_region_x, event.mouse_region_y)) self.mouse_path = [(self.mouse_prev, self.radius)] area = context.area region = context.region w, h = region.width, region.height rv3d = area.spaces.active.region_3d view_mat = rv3d.view_matrix.inverted() org = self.org = view_mat.to_translation() #org = region_2d_to_origin_3d(region, rv3d, (region.width/2.0, region.height/2.0)) #print('ORG', org) #print('view_mat', view_mat) self.cuts_data = [] #org = self.view_mat @ Vector((0, 0, -10)) #self.plane_no = self.plane_co-self.org #bottom_left = region_2d_to_location_3d(region, rv3d , (0, 0), self.plane_co) #bottom_right = region_2d_to_location_3d(region, rv3d , (0, w), self.plane_co) #bottom_left = intersect_line_plane(self.org, bottom_left, self.plane_co, self.plane_no) #bottom_right = intersect_line_plane(self.org, bottom_right, self.plane_co, self.plane_no) #self.scale_fac = (bottom_right-bottom_left).length / w #print('scale_fac', self.scale_fac) #depth_location = view_mat @ Vector((0, 0, -1)) #context.scene.cursor.location = depth_location #plane_2d = [(0, 0), (0, h), (w, h), (w, h)] #plane_3d = [region_2d_to_location_3d(p)] t0 = time() gp_mats = gp.data.materials gp_layers = [l for l in gp.data.layers if not l.lock or l.hide] self.gp_frames = [l.active_frame for l in gp_layers] ''' points_data = [(s, f, gp_mats[s.material_index]) for f in gp_frames for s in f.strokes] points_data = [(s, f, m) for s, f, m in points_data if not m.grease_pencil.hide or m.grease_pencil.lock] print('get_gp_points', time()-t0) t0 = time() #points_data = [(s, f, m, p, get_screen_co(p.co, matrix)) for s, f, m in points_data for p in reversed(s.points)] points_data = [(s, f, m, p, org + ((matrix @ p.co)-org).normalized()*1) for s, f, m in points_data for p in reversed(s.points)] print('points_to_2d', time()-t0) #print(points_data) self.points_data = [(s, f, m, p, co) for s, f, m, p, co in points_data if co is not None] #for s, f, m, p, co in self.points_data: # p.co = co t0 = time() self.kd_tree = KDTree(len(self.points_data)) for i, point_data in enumerate(self.points_data): s, f, m, p, co = point_data self.kd_tree.insert(co, i) self.kd_tree.balance() print('create kdtree', time()-t0) ''' ''' # Create holdout mat self.hld_mat = get_gp_mat(gp, name='Eraser Holdout Stroke') self.hld_mat.grease_pencil.use_stroke_holdout = True self.hld_mat.grease_pencil.show_stroke = True self.hld_mat.grease_pencil.show_fill = False self.hld_mat.grease_pencil.use_overlap_strokes = True self.hld_index = gp_mats[:].index(self.hld_mat) self.hld_strokes = [] for f in self.gp_frames: hld_stroke = f.strokes.new() hld_stroke.start_cap_mode = 'ROUND' hld_stroke.end_cap_mode = 'ROUND' hld_stroke.material_index = self.hld_index #hld_stroke.line_width = self.radius self.hld_strokes.append((f, hld_stroke)) self.draw_holdout(context, event) ''' context.area.tag_redraw() self._handle = bpy.types.SpaceView3D.draw_handler_add(self.draw_callback_px, (), 'WINDOW', 'POST_PIXEL') context.window_manager.modal_handler_add(self) return {'RUNNING_MODAL'} ### --- REGISTER --- classes=( GPTB_OT_eraser, ) def register(): for cls in classes: bpy.utils.register_class(cls) def unregister(): for cls in reversed(classes): bpy.utils.unregister_class(cls)