diff --git a/.gitattributes b/.gitattributes index fcb7fc35f7..481d33ac77 100644 --- a/.gitattributes +++ b/.gitattributes @@ -2810,6 +2810,7 @@ components/lazreport/images/designer/lrd_copy.png -text components/lazreport/images/designer/lrd_cut.png -text components/lazreport/images/designer/lrd_grid.png -text components/lazreport/images/designer/lrd_grid_align.png -text +components/lazreport/images/designer/lrd_guides.png -text components/lazreport/images/designer/lrd_ins_fields.bmp -text svneol=unset#image/x-ms-bmp components/lazreport/images/designer/lrd_italic.png -text components/lazreport/images/designer/lrd_new.png -text diff --git a/components/lazreport/images/designer/lrd_guides.png b/components/lazreport/images/designer/lrd_guides.png new file mode 100644 index 0000000000..cbf52e0d99 Binary files /dev/null and b/components/lazreport/images/designer/lrd_guides.png differ diff --git a/components/lazreport/source/lr_class.pas b/components/lazreport/source/lr_class.pas index 87e9de3f18..41817b1ad4 100644 --- a/components/lazreport/source/lr_class.pas +++ b/components/lazreport/source/lr_class.pas @@ -28,6 +28,7 @@ uses const lrMaxBandsInReport = 256; //temp fix. in future need remove this limit + lrSnapDistance: Integer = 10; const // object flags @@ -352,6 +353,7 @@ type procedure SetBounds(aLeft, aTop, aWidth, aHeight: Integer); function PointInView(aX,aY : Integer) : Boolean; virtual; + function FindAlignSide(const vert:boolean; const value: Integer; out found: Integer): boolean; virtual; procedure Invalidate; property Canvas : TCanvas read fCanvas write fCanvas; @@ -3281,6 +3283,29 @@ begin Result:=((aX>Rc.Left) and (aXRc.Top) and (aYnil) then + begin + r := Rect(px^-4, 0 , px^+4, fOwner.ClientHeight-1); + InvalidateRect(fOwner.Handle, @r, false); + end; +end; + +procedure TAlignGuides.InvalidateVertGuide; +var + r: TRect; +begin + if (py<>nil) then + begin + r := Rect(0, py^-4, fOwner.ClientWidth-1, py^+4); + InvalidateRect(fOwner.Handle, @r, false); + end; +end; + +procedure TAlignGuides.PaintGuides; +var + oldStyle: TPenStyle; + oldColor: TColor; + oldCosmetic: Boolean; + i, v, oldWidth: Integer; + t: TfrView; +begin + if (px<>nil) or (py<>nil) then + with fOwner.Canvas do + begin + oldStyle := Pen.Style; + oldColor := Pen.Color; + oldCosmetic := Pen.Cosmetic; + oldWidth := Pen.Width; + + // paint object's aligned sides + // TODO: make an option for the fixed values + + Pen.Cosmetic := true; + Pen.Style := psSolid; + Pen.Width := 5; + Pen.Color := clSkyBlue; + + for i:=0 to Objects.Count-1 do + begin + t := TfrView(Objects[i]); + if px<>nil then + if t.FindAlignSide(false, px^, v) and (v=px^) then + begin + MoveTo(px^, t.y); + LineTo(px^, t.y + t.dy); + end; + if py<>nil then + if t.FindAlignSide(true, py^, v) and (v=py^) then + begin + MoveTo(t.x, py^); + LineTo(t.x + t.dx, py^); + end; + end; + + // paint guides + // TODO: make an option for the fixed values + + Pen.Style := psDash; + Pen.Cosmetic := false; + Pen.Width := 1; + + if px<>nil then + begin + Pen.Color := clRed; + MoveTo(px^, 0); + LineTo(px^, fOwner.ClientHeight); + end; + if py<>nil then + begin + Pen.Color := clBlue; + MoveTo(0, py^); + LineTo(fOwner.ClientWidth, py^); + end; + + Pen.Cosmetic := oldCosmetic; + Pen.Style := oldStyle; + Pen.Color := oldColor; + Pen.Width := oldWidth; + end; +end; + +constructor TAlignGuides.Create(aOwner: TfrDesignerPage); +begin + inherited create; + fOwner := aOwner; +end; + +procedure TAlignGuides.Paint; +begin + PaintGuides; +end; + +procedure TAlignGuides.FindGuides(ax, ay: Integer); +var + i, tx, ty: Integer; + t: TfrView; + foundVert, foundHorz: boolean; +begin + foundVert := false; + foundHorz := false; + + // TODO: start looking at the nearest object to (ax, ay) + + for i := 0 to Objects.Count-1 do + begin + t := TfrView(Objects[i]); + + if not foundHorz and t.FindAlignSide(false, ax, tx) then + begin + if (px=nil) or (px^<>tx) then + begin + InvalidateHorzGuide; + fx := tx; + px := @fx; + InvalidateHorzGuide; + end; + foundHorz := true; + end; + + if not foundVert and t.FindAlignSide(true, ay, ty) then + begin + if (py=nil) or (py^<>ty) then + begin + InvalidateVertGuide; + fy := ty; + py := @fy; + InvalidateVertGuide; + end; + foundvert := true; + end; + + if foundHorz and foundVert then + break; + end; + + if not foundHorz and (px<>nil) then + begin + InvalidateHorzGuide; + px := nil; + end; + + if not foundVert and (py<>nil) then + begin + InvalidateVertGuide; + py := nil; + end; +end; + +function TAlignGuides.SnapToGuide(var ax, ay: Integer): boolean; +var + newX, newY: Integer; +begin + newX := ax; newY := ay; + if (px<>nil) and (Abs(ax-px^)<=lrSnapDistance) then + newX := px^; + if (py<>nil) and (Abs(ay-py^)<=lrSnapDistance) then + newY := py^; + result := (newX<>ax) or (newY<>ay); + if result then + begin + ax := newX; + ay := newY; + end; +end; + +procedure TAlignGuides.HideGuides; +begin + InvalidateHorzGuide; + InvalidateVertGuide; + px := nil; + py := nil; +end; + { TPaintSel } constructor TPaintSel.Create(AOwner: TfrDesignerPage); @@ -934,7 +1145,8 @@ begin Parent := AOwner as TWinControl; Color := clWhite; EnableEvents; - fPaintSel := TPaintSel.Create(self); + fPaintSel := TPaintSel.Create(self); + fGuides := TAlignGuides.Create(self); end; destructor TfrDesignerPage.destroy; @@ -981,6 +1193,7 @@ procedure TfrDesignerPage.Paint; begin fPainting := true; Draw(10000, 0); + fGuides.Paint; fPaintSel.Paint; fPainting := false; end; @@ -1519,11 +1732,17 @@ end; procedure TfrDesignerPage.RoundCoord(var x, y: Integer); begin with FDesigner do + begin + + if ShowGuides and fGuides.SnapToGuide(x, y) then + exit; + if GridAlign then begin x := x div GridSize * GridSize; y := y div GridSize * GridSize; end; + end; end; procedure TfrDesignerPage.GetMultipleSelected; @@ -1560,6 +1779,12 @@ begin end; end; +procedure TfrDesignerPage.CheckGuides; +begin + if not FDesigner.ShowGuides then + fGuides.HideGuides; +end; + procedure TfrDesignerPage.MDown(Sender: TObject; Button: TMouseButton; Shift: TShiftState; X, Y: Integer); var @@ -2214,6 +2439,9 @@ begin Moved := True; w := 2; + if FDesigner.ShowGuides then + fGuides.FindGuides(x, y); + if FirstChange and Down and not RFlag then begin kx := x - LastX; @@ -2437,13 +2665,19 @@ begin // sizing several objects if Down and TfrDesignerForm(frDesigner).MRFlag and (Mode = mdSelect) and (Cursor <> crDefault) then begin - kx := x - LastX; - ky := y - LastY; - if FDesigner.GridAlign and not GridCheck then begin - {$IFDEF DebugLR} - DebugLnExit('TfrDesignerPage.MMove DONE: sizing seveal, not gridcheck'); - {$ENDIF} - Exit; + + if FDesigner.ShowGuides and fGuides.SnapToGuide(x, y) then begin + kx := x - LastX; + ky := y - LastY; + end else begin + kx := x - LastX; + ky := y - LastY; + if FDesigner.GridAlign and not GridCheck then begin + {$IFDEF DebugLR} + DebugLnExit('TfrDesignerPage.MMove DONE: sizing seveal, not gridcheck'); + {$ENDIF} + Exit; + end; end; if FDesigner.ShapeMode = smFrame then @@ -2876,6 +3110,7 @@ begin NPEraseFocusRect; OffsetRect(OldRect, -10000, -10000); end; + fGuides.HideGuides; end; {-----------------------------------------------------------------------------} @@ -3532,6 +3767,11 @@ begin ViewsAction(nil, @ToggleFrames, -1); end; +procedure TfrDesignerForm.btnGuidesClick(Sender: TObject); +begin + ShowGuides := btnGuides.Down; +end; + procedure TfrDesignerForm.FormShow(Sender: TObject); var CursorImage: TCursorImage; @@ -3698,6 +3938,14 @@ begin FGridAlign := Value; end; +procedure TfrDesignerForm.SetGuidesShow(AValue: boolean); +begin + if FGuidesShow = AValue then Exit; + FGuidesShow := AValue; + btnGuides.Down := AValue; + PageView.CheckGuides; +end; + procedure TfrDesignerForm.SetUnits(Value: TfrReportUnits); var s: String; @@ -6505,6 +6753,7 @@ const rsGridShow = 'GridShow'; rsGridAlign = 'GridAlign'; rsGridSize = 'GridSize'; + rsGuidesShow = 'GuidesShow'; rsUnits = 'Units'; rsButtons = 'GrayButtons'; rsEdit = 'EditAfterInsert'; @@ -6534,6 +6783,7 @@ begin Ini.WriteBool('frEditorForm', rsGridShow, ShowGrid); Ini.WriteBool('frEditorForm', rsGridAlign, GridAlign); + Ini.WriteBool('frEditorForm', rsGuidesShow, ShowGuides); Ini.WriteInteger('frEditorForm', rsGridSize, GridSize); Ini.WriteInteger('frEditorForm', rsUnits, Word(Units)); Ini.WriteBool('frEditorForm', rsButtons, GrayedButtons); @@ -6584,6 +6834,7 @@ begin GridSize := Ini.ReadInteger('frEditorForm', rsGridSize, 4); GridAlign := Ini.ReadBool('frEditorForm', rsGridAlign, True); ShowGrid := Ini.ReadBool('frEditorForm', rsGridShow, True); + ShowGuides := Ini.ReadBool('frEditorForm', rsGuidesShow, true); Units := TfrReportUnits(Ini.ReadInteger('frEditorForm', rsUnits, 0)); // GrayedButtons := Ini.ReadBool('frEditorForm', rsButtons, False); EditAfterInsert := Ini.ReadBool('frEditorForm', rsEdit, True);