| 1 | Recently was asked by a user that the Zoom to Rectangle tool in Mapguide doesn't work. In investigating their statement, it turns out that when they were using the 'Zoom to Rectangle' tool they were drawing a rectangle that went precisely over top of the object of interest, and therefore was being cut off on the outer bounds. |
| 2 | |
| 3 | In looking at the code base for the 'Zoom to Rectangle' function calculates the new mapscale based on the largest side of the rectangle, AND assumes the first x/y position of the rectangle is the top left (90% of the time it would be but some people will draw a rectangle from lower right to top left). |
| 4 | |
| 5 | The following code adjusts the 'Zoom to Rectangle' function slightly so that no matter what order the rectangle is captured in (lower right to top left/top right to lower left/top left to lower right/etc), it resets the x1/y1 values to be the top left, and recalculates the shape of the zoom rectangle to roughly fit the shape of the mapframe. It does this by getting the larger edge of the zoom rectangle, and adjust the other edge by applying the ratio of width to height of the mapframe. In addition to the recalculate of the zoom rectangle shape, it puts a expands it by 5px, to ensure that the object being zoomed to fits in the mapframe and is not cut off. |
| 6 | |
| 7 | To use this code replace the ExecuteRectangleTool() function in ajaxmappane.templ, with the one below. |
| 8 | |
| 9 | {{{ |
| 10 | function ExecuteRectangleTool() |
| 11 | { |
| 12 | var x1, x2; |
| 13 | var y1, y2; |
| 14 | |
| 15 | if(rcx1 <= rcx2) |
| 16 | { |
| 17 | x1 = rcx1; |
| 18 | x2 = rcx2; |
| 19 | } |
| 20 | else |
| 21 | { |
| 22 | x1 = rcx2; |
| 23 | x2 = rcx1; |
| 24 | } |
| 25 | |
| 26 | if(rcy1 <= rcy2) |
| 27 | { |
| 28 | y1 = rcy1; |
| 29 | y2 = rcy2; |
| 30 | } |
| 31 | else |
| 32 | { |
| 33 | y1 = rcy2; |
| 34 | y2 = rcy1; |
| 35 | } |
| 36 | |
| 37 | if(tool == 3) |
| 38 | { |
| 39 | if((x2 - x1 <= 2) && (y2 - y1 <= 2)) |
| 40 | return; |
| 41 | |
| 42 | //ensure we have the box extents as true upper left/lower right |
| 43 | var minx = Math.min(x1, x2); |
| 44 | var miny = Math.min(y1, y2); |
| 45 | var maxx = Math.max(x1, x2); |
| 46 | var maxy = Math.max(y1, y2); |
| 47 | |
| 48 | |
| 49 | //get the width and height of the zoom box/window |
| 50 | var width = maxx - minx; |
| 51 | var height = maxy - miny; |
| 52 | |
| 53 | //get the ratio of width to height from the mapframe, this is to ensure the dimensions of the zoom box best fit the mapframe. |
| 54 | var ratio = mapDevW / mapDevH; |
| 55 | |
| 56 | //deterimine whether the width or height is to be reset. |
| 57 | if (width < height) |
| 58 | { |
| 59 | width = ratio * height; |
| 60 | } |
| 61 | else |
| 62 | { |
| 63 | height = width / ratio; |
| 64 | } |
| 65 | |
| 66 | x2 = minx + width; |
| 67 | y2 = miny + height; |
| 68 | |
| 69 | |
| 70 | // add a 5 px margin to the zoom box to ensure coverage of area of interest. |
| 71 | x1 = minx - 5; |
| 72 | y1 = miny - 5; |
| 73 | x2 = x2 + 5; |
| 74 | y2 = y2 + 5; |
| 75 | |
| 76 | |
| 77 | pt1 = ScreenToMapUnits(x1, y1); |
| 78 | pt2 = ScreenToMapUnits(x2, y2); |
| 79 | |
| 80 | mcsW = pt2.X - pt1.X; |
| 81 | mcsH = pt2.Y - pt1.Y; |
| 82 | scale = CalculateScale1(mcsW, mcsH, mapDevW, mapDevH); |
| 83 | scale = finscale? GetNearestFiniteScale(scale): NormalizeScale(scale); |
| 84 | HideMapImage(); |
| 85 | GotoView(pt1.X + mcsW/2, pt1.Y + mcsH/2, scale, true, false); |
| 86 | } |
| 87 | else if(tool == 0) |
| 88 | { |
| 89 | if((x2 - x1 <= 2) && (y2 - y1 <= 2)) |
| 90 | { |
| 91 | pt1 = ScreenToMapUnits(x1-2, y1-2); |
| 92 | pt2 = ScreenToMapUnits(x1+2, y1+2); |
| 93 | RequestPointSelection(pt1.X, pt1.Y, pt2.X, pt2.Y, appending); |
| 94 | } |
| 95 | else |
| 96 | { |
| 97 | pt1 = ScreenToMapUnits(x1, y1); |
| 98 | pt2 = ScreenToMapUnits(x2, y2); |
| 99 | RequestRectSelection(pt1.X, pt1.Y, pt2.X, pt2.Y, appending); |
| 100 | } |
| 101 | } |
| 102 | } |
| 103 | }}} |
| 104 | |
| 105 | Comments are appreciated. |
| 106 | Jamie |