|
|
|
@ -1,6 +1,9 @@
|
|
|
|
|
import math |
|
|
|
|
import FreeCADGui as Gui |
|
|
|
|
import FreeCAD as App |
|
|
|
|
|
|
|
|
|
import ahb_utils |
|
|
|
|
|
|
|
|
|
class AHB_View_Annotate: |
|
|
|
|
def GetResources(self): |
|
|
|
|
return {"MenuText": "Annotate view", |
|
|
|
@ -67,9 +70,75 @@ class AHB_View_Annotate:
|
|
|
|
|
doc.removeObject(balloon.Name) |
|
|
|
|
|
|
|
|
|
for partLink in view.XSource: |
|
|
|
|
workbench.techDrawExtensions.add_or_update_balloon(view, partLink, '') |
|
|
|
|
|
|
|
|
|
balloonsCreated = workbench.techDrawExtensions.add_or_update_balloon(view, partLink, '') |
|
|
|
|
if len(balloonsCreated) > 0: |
|
|
|
|
regroupedBalloons = self.RegroupNearestSimilarBalloons(balloonsCreated) |
|
|
|
|
self.PlaceBalloonsInCircle(regroupedBalloons) |
|
|
|
|
#self.PlaceBalloonsInCircle(balloonsCreated) |
|
|
|
|
|
|
|
|
|
workbench.techDrawExtensions.refreshOverlays(page) |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def CalculatePointsCenter(self, balloons): |
|
|
|
|
totalX = 0 |
|
|
|
|
totalY = 0 |
|
|
|
|
for balloon in balloons: |
|
|
|
|
realBalloon = balloon[0] if type(balloon) is list else balloon |
|
|
|
|
totalX = totalX + int(realBalloon.OriginX) |
|
|
|
|
totalY = totalY + int(realBalloon.OriginY) |
|
|
|
|
return App.Base.Vector2d(totalX / len(balloons), totalY / len(balloons)) |
|
|
|
|
|
|
|
|
|
def IsSimilarBalloonNear(self, balloonA, balloonB): |
|
|
|
|
MAX_DISTANCE_BETWEEN_REGROUPED_BALLOONS = 50 |
|
|
|
|
if balloonA.Text == balloonB.Text: |
|
|
|
|
pos = App.Base.Vector2d(balloonA.OriginX, balloonA.OriginY) |
|
|
|
|
dist = pos.distance(App.Base.Vector2d(balloonB.OriginX, balloonB.OriginY)) |
|
|
|
|
return dist < MAX_DISTANCE_BETWEEN_REGROUPED_BALLOONS |
|
|
|
|
else: |
|
|
|
|
return False |
|
|
|
|
|
|
|
|
|
def RegroupNearestSimilarBalloons(self, balloons): |
|
|
|
|
regroupedBalloons = [] |
|
|
|
|
for balloon in balloons: |
|
|
|
|
nearestBalloons = [] |
|
|
|
|
for otherBalloon in balloons: |
|
|
|
|
if otherBalloon != balloon and self.IsSimilarBalloonNear(balloon, otherBalloon): |
|
|
|
|
nearestBalloons.append(otherBalloon) |
|
|
|
|
balloons.remove(otherBalloon) |
|
|
|
|
if len(nearestBalloons) == 0: |
|
|
|
|
regroupedBalloons.append(balloon) |
|
|
|
|
else: |
|
|
|
|
nearestBalloons.append(balloon) |
|
|
|
|
regroupedBalloons.append(nearestBalloons) |
|
|
|
|
return regroupedBalloons |
|
|
|
|
|
|
|
|
|
def PlaceBalloonsInCircle(self, balloons): |
|
|
|
|
center = self.CalculatePointsCenter(balloons) |
|
|
|
|
nbBalloons = len(balloons) |
|
|
|
|
balloonPosStep = (math.pi * 2) / nbBalloons |
|
|
|
|
for i in range(nbBalloons): |
|
|
|
|
xPos = round(center.x + 600 * math.cos(balloonPosStep * i)) |
|
|
|
|
yPos = round(center.y + 600 * math.sin(balloonPosStep * i)) |
|
|
|
|
balloonPos = App.Base.Vector2d(xPos, yPos) |
|
|
|
|
# Find nearest arrow to avoid arrow crossing each other |
|
|
|
|
smallestDistance = 0 |
|
|
|
|
balloonToUse = None |
|
|
|
|
for balloon in balloons: |
|
|
|
|
realBalloon = balloon[0] if type(balloon) is list else balloon |
|
|
|
|
dist = balloonPos.distance(App.Base.Vector2d(realBalloon.OriginX, realBalloon.OriginY)) |
|
|
|
|
if smallestDistance == 0 or dist < smallestDistance: |
|
|
|
|
smallestDistance = dist |
|
|
|
|
balloonToUse = balloon |
|
|
|
|
|
|
|
|
|
if balloonToUse is not None: |
|
|
|
|
balloons.remove(balloonToUse) |
|
|
|
|
if type(balloonToUse) is list: |
|
|
|
|
for realBalloon in balloonToUse: |
|
|
|
|
realBalloon.X = balloonPos.x |
|
|
|
|
realBalloon.Y = balloonPos.y |
|
|
|
|
else: |
|
|
|
|
balloonToUse.X = balloonPos.x |
|
|
|
|
balloonToUse.Y = balloonPos.y |
|
|
|
|
|
|
|
|
|
from ahb_command import AHB_CommandWrapper |
|
|
|
|
AHB_CommandWrapper.addGuiCommand('AHB_view_annotate', AHB_View_Annotate()) |
|
|
|
|