import java.awt.geom.Line2D ;
import java.io.BufferedReader ;
import java.io.IOException ;
import java.io.InputStreamReader ;
import java.util.ArrayList ;
import java.util.LinkedList ;
import java.util.List ;
import static java.
lang .
Double .
max ; import static java.
lang .
Double .
min ;
public class Main {
private static final double G = 9.80665 ;
static class Building {
int x, y, z;
boolean visited;
int steps = 0 ;
int i, j;
Building( int x, int y, int z, int i, int j) {
this .x = x;
this .y = y;
this .z = z;
this .i = i;
this .j = j;
}
@Override
return "Building{" +
"x=" + x +
", y=" + y +
", z=" + z +
'}' ;
}
}
double x1, y1, x2, y2;
public Segment ( double x1,
double y1,
double x2,
double y2
) { this .x1 = x1;
this .y1 = y1;
this .x2 = x2;
this .y2 = y2;
}
@Override
return "(" + x1 + ", " + y1 + ") -> (" + x2 + ", " + y2 + ")" ;
}
// https://g...content-available-to-author-only...b.com/baobabKoodaa/CodeForcesLibrary/blob/ad6972b1ccd87c9100171122692562704c695703/src/baobab/A.java#L767
// Returns true if segment 1-2 intersects segment 3-4
double x3 = s.x1 , x4 = s.x2 , y3 = s.y1 , y4 = s.y2 ;
if ( x1 == x2 && x3 == x4) {
// Both segments are vertical
if ( x1 != x3) {
return null ;
}
if ( min( y1, y2) < min( y3, y4) ) {
if ( max( y1, y2) >= min( y3, y4) ) {
return OVERLAP;
} else {
return null ;
}
} else {
if ( max( y3, y4) >= min( y1, y2) ) {
return OVERLAP;
} else {
return null ;
}
}
}
if ( x1 == x2) {
// Only segment 1-2 is vertical. Does segment 3-4 cross it? y = a*x + b
double a34 = ( y4 - y3) / ( x4 - x3) ;
double b34 = y3 - a34 * x3;
double y = a34 * x1 + b34;
if ( y >= min( y1, y2) - 1e- 6 && y <= max( y1, y2) + 1e- 6 && x1 >= min( x3, x4) - 1e- 6 && x1 <= max( x3, x4) + 1e- 6) {
} else {
return null ;
}
}
if ( x3 == x4) {
// Only segment 3-4 is vertical. Does segment 1-2 cross it? y = a*x + b
double a12 = ( y2 - y1) / ( x2 - x1) ;
double b12 = y1 - a12 * x1;
double y = a12 * x3 + b12;
if ( y >= min( y3, y4) - 1e- 6 && y <= max( y3, y4) + 1e- 6 && x3 >= min( x1, x2) - 1e- 6 && x3 <= max( x1, x2) + 1e- 6) {
} else {
return null ;
}
}
double a12 = ( y2 - y1) / ( x2 - x1) ;
double b12 = y1 - a12 * x1;
double a34 = ( y4 - y3) / ( x4 - x3) ;
double b34 = y3 - a34 * x3;
if ( closeToZero( a12 - a34) ) {
// Parallel lines
if ( closeToZero( b12 - b34) ) {
return OVERLAP;
} else {
return null ;
}
}
// Non parallel non vertical lines intersect at x. Is x part of both segments?
double x = - ( b12 - b34) / ( a12 - a34) ;
if ( x >= min( x1, x2) - 1e- 6 && x <= max( x1, x2) + 1e- 6 && x >= min( x3, x4) - 1e- 6 && x <= max( x3, x4) + 1e- 6) {
return new Point ( x, a12
* x
+ b12
) ; } else {
return null ;
}
}
boolean closeToZero( double v) {
// Check if double is close to zero, considering precision issues.
return Math .
abs ( v
) <= 0.0000000000001 ; }
}
double x, y;
public Point ( double x,
double y
) { this .x = x;
this .y = y;
}
@Override
return "(" + x + ", " + y + ")" ;
}
}
int [ ] v = readArrayLine( br.readLine ( ) , 6 ) ;
int m = v[ 0 ] , n = v[ 1 ] ;
double w = v[ 2 ] , speed = v[ 3 ] ;
int lx = v[ 4 ] - 1 , ly = v[ 5 ] - 1 ;
Building[ ] [ ] b = new Building[ n] [ m] ;
int currentX;
int currentY = 0 ;
for ( int i = 0 ; i < n; i++ ) {
v = readArrayLine( br.readLine ( ) , m) ;
currentX = 0 ;
for ( int j = 0 ; j < m; j++ ) {
b[ i] [ j] = new Building( currentX, currentY, v[ j] , i, j) ;
currentX += w;
}
currentY += w;
}
// Segment[] horizontal = new Segment[n - 1];
// Segment[] vertical = new Segment[m - 1];
// for (int i = 0; i < n - 1; i++) {
// double hy = w / 2 + w * i;
// horizontal[i] = new Segment(0, hy, w * (n - 1), hy);
// }
//
// for (int i = 0; i < m - 1; i++) {
// double vx = w / 2 + w * i;
// vertical[i] = new Segment(vx, 0, vx, w * (m - 1));
// }
Building root = b[ ly] [ lx] ;
LinkedList< Building> queue = new LinkedList<> ( ) ;
queue.add ( root) ;
root.visited = true ;
while ( ! queue.isEmpty ( ) ) {
root = queue.poll ( ) ;
if ( root == null ) {
break ;
}
for ( int i = 0 ; i < n; i++ ) {
for ( int j = 0 ; j < m; j++ ) {
Building dest = b[ i] [ j] ;
if ( dest.visited ) {
continue ;
}
// if (canJump(root, dest, speed, vertical, horizontal, w, b)) {
if ( canJumpSmart( root, dest, speed, w, b) ) {
dest.steps = root.steps + 1 ;
dest.visited = true ;
queue.add ( dest) ;
}
}
}
}
for ( int i = 0 ; i < n; i++ ) {
for ( int j = 0 ; j < m; j++ ) {
if ( b[ i] [ j] .steps == 0 && ( i != ly || j != lx) ) {
} else {
System .
out .
print ( b
[ i
] [ j
] .
steps + " " ) ; }
}
}
}
// private static boolean canJump(Building root, Building dest, double speed, Segment[] vertical, Segment[] horizontal, double w, Building b[][]) {
// double dist = getHorizontalDistBetweenSquared(root, dest);
// int heightDiff = dest.z - root.z;
//
// double launchAngle = getAngle(speed, Math.sqrt(dist), heightDiff);
// if (launchAngle > 0) {
// return checkTrajectory(b, speed, root, dest, Math.tan(launchAngle), vertical, horizontal, w);
// }
// return false;
// }
private static boolean canJumpSmart( Building root, Building dest, double speed, double w, Building b[ ] [ ] ) {
double dist = getHorizontalDistBetweenSquared( root, dest) ;
int heightDiff = dest.z - root.z ;
double tangentOfLaunchAngle
= getTangentOfLaunchAngle
( speed,
Math .
sqrt ( dist
) , heightDiff,
1 ) ; double tangentOfLaunchAngle1
= getTangentOfLaunchAngle
( speed,
Math .
sqrt ( dist
) , heightDiff,
- 1 ) ;
boolean check1 = false ;
if ( tangentOfLaunchAngle > 0 ) {
check1 = checkTrajectorySimple( b, speed, root, dest,tangentOfLaunchAngle, w) ;
}
boolean check2 = false ;
if ( tangentOfLaunchAngle1 > 0 ) {
check2 = checkTrajectorySimple( b, speed, root, dest, tangentOfLaunchAngle1, w) ;
}
return check1 || check2;
}
private static boolean isAboveBuilding
( Building b,
double speed,
double tangentOfLaunchAngle,
Point intersection,
int h0
) { double xDist = b.x - intersection.x ;
double yDist = b.y - intersection.y ;
double dist
= Math .
sqrt ( xDist
* xDist
+ yDist
* yDist
) ; double computedHeight = computeHeightForDistAndTan( h0, speed, tangentOfLaunchAngle, dist) ;
return computedHeight >= b.z - 1e- 9;
}
private static boolean checkTrajectorySimple( Building[ ] [ ] b, double speed, Building root, Building dest, double tan, double w) {
int h0 = root.z ;
for ( int i = 0 ; i < b.length ; i ++ ) {
for ( int j = 0 ; j < b[ i] .length ; j++ ) {
double x = b[ i] [ j] .x ;
double y = b[ i] [ j] .y ;
List< Point> polygon = new ArrayList<> ( ) ;
polygon.
add ( new Point ( x
- w
/ 2 , y
- w
/ 2 ) ) ; polygon.
add ( new Point ( x
+ w
/ 2 , y
- w
/ 2 ) ) ; polygon.
add ( new Point ( x
+ w
/ 2 , y
+ w
/ 2 ) ) ; polygon.
add ( new Point ( x
- w
/ 2 , y
+ w
/ 2 ) ) ;
for ( int k = 0 ; k < 4 ; k ++ ) {
Point p1
= polygon.
get ( k
) ; Point p2
= polygon.
get ( ( k
+ 1 ) % polygon.
size ( ) ) ; intersect = s.intersect ( polySeg) ;
boolean inter
= Line2D .
linesIntersect ( p1.
x , p1.
y , p2.
x , p2.
y , s.
x1 , s.
y1 , s.
x2 , s.
y2 ) ; if ( intersect
!= null && intersect
!= Segment .
OVERLAP ) { if ( ! inter) {
throw null ;
}
if ( ! isAboveBuilding( b[ i] [ j] , speed, tan, intersect, h0) ) {
return false ;
}
} else {
if ( inter) {
// System.out.println(s + " " + polySeg + " " + intersect);
throw null ;
}
}
}
}
}
return true ;
}
// private static boolean checkTrajectory(Building[][] b, double speed, Building root, Building dest, double tan, Segment[] verticals, Segment[] horizontals, double w) {
//
// Segment s = new Segment(root.x, root.y, dest.x, dest.y);
// Map<Segment, Point> segmentPointMap = new HashMap<>();
// int h0 = root.z;
// for (int i = 0; i < horizontals.length; i++) {
// Segment segment = horizontals[i];
// Point intersect = s.intersect(segment);
// if (intersect != null && intersect != Segment.OVERLAP) {
//// System.out.println(intersect + " = " + ("[" + root.i + ", " + root.j + "]") + " " + ("[" + dest.i + ", " + dest.j + "]") + " ----> " + segment);
// segmentPointMap.put(segment, intersect);
// if (verticals.length > 0) {
// if (intersect.x >= verticals[0].x1 - w && intersect.x <= verticals[0].x1) {
//// System.out.println(("-> [" + i + ", " + 0 + "]") + " " + ("[" + (i + 1) + ", " + 0 + "]"));
// if (!isAboveBuilding(b[i][0], speed, tan, intersect, h0)
// || !isAboveBuilding(b[i + 1][0], speed, tan, intersect, h0)) {
// return false;
// }
// }
// for (int j = 0; j < verticals.length; j++) {
// if (intersect.x >= verticals[j].x1 && intersect.x <= verticals[j].x1 + w) {
//// System.out.println(("-> [" + i + ", " + (j + 1) + "]") + " " + ("[" + (i + 1) + ", " + (j + 1) + "]"));
// if (!isAboveBuilding(b[i][j + 1], speed, tan, intersect, h0)
// || !isAboveBuilding(b[i + 1][j + 1], speed, tan, intersect, h0)) {
// return false;
// }
// }
// }
// } else {
//// System.out.println(("-> [" + i + ", " + 0 + "]") + " " + ("[" + (i + 1) + ", " + 0 + "]"));
// if (!isAboveBuilding(b[i][0], speed, tan, intersect, h0)
// || !isAboveBuilding(b[i + 1][0], speed, tan, intersect, h0)) {
// return false;
// }
// }
// }
// }
//
// for (int i = 0; i < verticals.length; i++) {
// Segment segment = verticals[i];
// Point intersect = s.intersect(segment);
// if (intersect != null && intersect != Segment.OVERLAP) {
//// System.out.println(intersect + " = " + ("[" + root.i + ", " + root.j + "]") + " " + ("[" + dest.i + ", " + dest.j + "]") + " ----> " + segment);
// segmentPointMap.put(segment, intersect);
// if (horizontals.length > 0) {
// if (intersect.y >= horizontals[0].y1 - w && intersect.y <= horizontals[0].y1) {
//// System.out.println(("-< [" + 0 + ", " + i + "]") + " " + ("[" + 0 + ", " + (i + 1) + "]"));
// if (!isAboveBuilding(b[0][i], speed, tan, intersect, h0)
// || !isAboveBuilding(b[0][i + 1], speed, tan, intersect, h0)) {
// return false;
// }
// }
// for (int j = 0; j < horizontals.length; j++) {
// if (intersect.y >= horizontals[j].y1 && intersect.y <= horizontals[j].y1 + w) {
//// System.out.println(("-< [" + (j + 1) + ", " + i + "]") + " " + ("[" + (j + 1) + ", " + (i + 1) + "]"));
// if (!isAboveBuilding(b[j + 1][i], speed, tan, intersect, h0)
// || !isAboveBuilding(b[j + 1][i + 1], speed, tan, intersect, h0)) {
// return false;
// }
// }
// }
// } else {
//// System.out.println(("-< [" + 0 + ", " + i + "]") + " " + ("[" + 0 + ", " + (i + 1) + "]"));
// if (!isAboveBuilding(b[0][i], speed, tan, intersect, h0)
// || !isAboveBuilding(b[0][i + 1], speed, tan, intersect, h0)) {
// return false;
// }
// }
// }
// }
//
//// System.out.println("end for " + ("[" + root.i + ", " + root.j + "]") + " " + ("[" + dest.i + ", " + dest.j + "]"));
//// System.out.println();
// return true;
// }
private static double getHorizontalDistBetweenSquared( Building b1, Building b2) {
return ( b1.x - b2.x ) * ( b1.x - b2.x ) + ( b1.y - b2.y ) * ( b1.y - b2.y ) ;
}
private static double getTangentOfLaunchAngle( double speed, double dist, double h, int coef) {
double a = G * dist * dist / ( 2 * speed * speed) ;
double b = - dist;
double c = a + h;
double delta = b * b - 4 * a * c;
if ( delta < 0 ) {
return - 1 ;
}
return ( - b
+ coef
* Math .
sqrt ( delta
) ) / ( 2 * a
) ; }
private static double getAngle( double v, double dist, double h) {
double start
= Math .
PI / 2 ; double end
= Math .
PI / 4 ; double computedY
= Double .
MAX_VALUE ; double x = 0 ;
while ( Math .
abs ( computedY
- h
) > 0.000005 ) { x = ( start + end) / 2 ;
if ( start == x || end == x) {
return - 1 ;
}
double vy
= v
* Math .
sin ( x
) ; double timeToHighest = vy / G;
double vx
= v
* Math .
cos ( x
) ; double timeToDist = dist / vx;
if ( timeToDist < timeToHighest - 1e- 9) {
end = x;
continue ;
}
computedY
= computeHeightForDistAndTan
( 0 , v,
Math .
tan ( x
) , dist
) ; if ( start - end < 0.00000000000000000000001 ) {
return - 1 ;
}
if ( computedY < h) {
start = x;
} else {
end = x;
}
}
return x;
}
private static double computeHeightForDistAndTan( int h0, double speed, double tan, double dist) {
return h0 + dist * tan - G * dist * dist * ( 1 + tan * tan) / ( 2 * speed * speed) ;
}
public static int [ ] readArrayLine
( String line,
int n
) { return readArrayLine( line, n, null , 0 ) ;
}
private static int [ ] readArrayLine
( String line,
int n,
int array
[ ] ,
int pos
) { int [ ] ret = array == null ? new int [ n] : array;
int start = 0 ;
int end = line.indexOf ( ' ' , start) ;
for ( int i = pos; i < pos + n; i++ ) {
if ( end > 0 )
ret
[ i
] = Integer .
parseInt ( line.
substring ( start, end
) ) ; else
ret
[ i
] = Integer .
parseInt ( line.
substring ( start
) ) ; start = end + 1 ;
end = line.indexOf ( ' ' , start) ;
}
return ret;
}
}


import java.awt.geom.Line2D;
import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.util.ArrayList;
import java.util.LinkedList;
import java.util.List;

import static java.lang.Double.max;
import static java.lang.Double.min;

public class Main {

    private static final double G = 9.80665;

    static class Building {
        int x, y, z;
        boolean visited;
        int steps = 0;
        int i, j;

        Building(int x, int y, int z, int i, int j) {
            this.x = x;
            this.y = y;
            this.z = z;
            this.i = i;
            this.j = j;
        }

        @Override
        public String toString() {
            return "Building{" +
                    "x=" + x +
                    ", y=" + y +
                    ", z=" + z +
                    '}';
        }
    }

    static class Segment {
        double x1, y1, x2, y2;

        public Segment(double x1, double y1, double x2, double y2) {
            this.x1 = x1;
            this.y1 = y1;
            this.x2 = x2;
            this.y2 = y2;
        }

        @Override
        public String toString() {
            return "(" + x1 + ", " + y1 + ") -> (" + x2 + ", " + y2 + ")";
        }

        //        https://g...content-available-to-author-only...b.com/baobabKoodaa/CodeForcesLibrary/blob/ad6972b1ccd87c9100171122692562704c695703/src/baobab/A.java#L767
        static final Point OVERLAP = new Point(0, 0);

        Point intersect(Segment s) {
            // Returns true if segment 1-2 intersects segment 3-4
            double x3 = s.x1, x4 = s.x2, y3 = s.y1, y4 = s.y2;

            if (x1 == x2 && x3 == x4) {
                // Both segments are vertical
                if (x1 != x3) {
                    return null;
                }
                if (min(y1, y2) < min(y3, y4)) {
                    if (max(y1, y2) >= min(y3, y4)) {
                        return OVERLAP;
                    } else {
                        return null;
                    }
                } else {
                    if (max(y3, y4) >= min(y1, y2)) {
                        return OVERLAP;
                    } else {
                        return null;
                    }
                }
            }
            if (x1 == x2) {
                // Only segment 1-2 is vertical. Does segment 3-4 cross it? y = a*x + b
                double a34 = (y4 - y3) / (x4 - x3);
                double b34 = y3 - a34 * x3;
                double y = a34 * x1 + b34;
                if (y >= min(y1, y2) - 1e-6 && y <= max(y1, y2)  + 1e-6 && x1 >= min(x3, x4) - 1e-6 && x1 <= max(x3, x4) + 1e-6) {
                    return new Point(x1, y);
                } else {
                    return null;
                }
            }
            if (x3 == x4) {
                // Only segment 3-4 is vertical. Does segment 1-2 cross it? y = a*x + b
                double a12 = (y2 - y1) / (x2 - x1);
                double b12 = y1 - a12 * x1;
                double y = a12 * x3 + b12;
                if (y >= min(y3, y4) - 1e-6 && y <= max(y3, y4) + 1e-6 && x3 >= min(x1, x2) - 1e-6 && x3 <= max(x1, x2) + 1e-6) {
                    return new Point(x3, y);
                } else {
                    return null;
                }
            }
            double a12 = (y2 - y1) / (x2 - x1);
            double b12 = y1 - a12 * x1;
            double a34 = (y4 - y3) / (x4 - x3);
            double b34 = y3 - a34 * x3;
            if (closeToZero(a12 - a34)) {
                // Parallel lines
                if (closeToZero(b12 - b34)) {
                    return OVERLAP;
                } else {
                    return null;
                }
            }
            // Non parallel non vertical lines intersect at x. Is x part of both segments?
            double x = -(b12 - b34) / (a12 - a34);
            if (x >= min(x1, x2) - 1e-6 && x <= max(x1, x2) + 1e-6 && x >= min(x3, x4) - 1e-6 && x <= max(x3, x4) + 1e-6) {
                return new Point(x, a12 * x + b12);
            } else {
                return null;
            }
        }

        boolean closeToZero(double v) {
            // Check if double is close to zero, considering precision issues.
            return Math.abs(v) <= 0.0000000000001;
        }
    }

    static class Point {
        double x, y;

        public Point(double x, double y) {
            this.x = x;
            this.y = y;
        }

        @Override
        public String toString() {
            return "(" + x + ", " + y + ")";
        }
    }

    public static void main(String[] args) throws IOException {

        BufferedReader br = new BufferedReader(new InputStreamReader(System.in), 1024 * 1024 * 2);

        int[] v = readArrayLine(br.readLine(), 6);
        int m = v[0], n = v[1];
        double w = v[2], speed = v[3];
        int lx = v[4] - 1, ly = v[5] - 1;

        Building[][] b = new Building[n][m];

        int currentX;
        int currentY = 0;
        for (int i = 0; i < n; i++) {
            v = readArrayLine(br.readLine(), m);
            currentX = 0;
            for (int j = 0; j < m; j++) {
                b[i][j] = new Building(currentX, currentY, v[j], i, j);
                currentX += w;
            }
            currentY += w;
        }

//        Segment[] horizontal = new Segment[n - 1];
//        Segment[] vertical = new Segment[m - 1];
//        for (int i = 0; i < n - 1; i++) {
//            double hy = w / 2 + w * i;
//            horizontal[i] = new Segment(0, hy, w * (n - 1), hy);
//        }
//
//        for (int i = 0; i < m - 1; i++) {
//            double vx = w / 2 + w * i;
//            vertical[i] = new Segment(vx, 0, vx, w * (m - 1));
//        }

        Building root = b[ly][lx];
        LinkedList<Building> queue = new LinkedList<>();
        queue.add(root);
        root.visited = true;

        while (!queue.isEmpty()) {
            root = queue.poll();

            if (root == null) {
                break;
            }

            for (int i = 0; i < n; i++) {
                for (int j = 0; j < m; j++) {
                    Building dest = b[i][j];
                    if (dest.visited) {
                        continue;
                    }

//                    if (canJump(root, dest, speed, vertical, horizontal, w, b)) {
                    if (canJumpSmart(root, dest, speed, w, b)) {
                        dest.steps = root.steps + 1;
                        dest.visited = true;
                        queue.add(dest);
                    }
                }
            }
        }

        for (int i = 0; i < n; i++) {
            for (int j = 0; j < m; j++) {
                if (b[i][j].steps == 0 && (i != ly || j != lx)) {
                    System.out.print("X ");
                } else {
                    System.out.print(b[i][j].steps + " ");
                }
            }
            System.out.println();
        }
    }


//    private static boolean canJump(Building root, Building dest, double speed, Segment[] vertical, Segment[] horizontal, double w, Building b[][]) {
//        double dist = getHorizontalDistBetweenSquared(root, dest);
//        int heightDiff = dest.z - root.z;
//
//        double launchAngle = getAngle(speed, Math.sqrt(dist), heightDiff);
//        if (launchAngle > 0) {
//            return checkTrajectory(b, speed, root, dest, Math.tan(launchAngle), vertical, horizontal, w);
//        }
//        return false;
//    }

    private static boolean canJumpSmart(Building root, Building dest, double speed, double w, Building b[][]) {
        double dist = getHorizontalDistBetweenSquared(root, dest);
        int heightDiff = dest.z - root.z;

        double tangentOfLaunchAngle = getTangentOfLaunchAngle(speed, Math.sqrt(dist), heightDiff, 1);
        double tangentOfLaunchAngle1 = getTangentOfLaunchAngle(speed, Math.sqrt(dist), heightDiff, -1);

        boolean check1 = false;
        if (tangentOfLaunchAngle > 0) {
            check1 = checkTrajectorySimple(b, speed, root, dest,tangentOfLaunchAngle, w);
        }

        boolean check2 = false;
        if (tangentOfLaunchAngle1 > 0) {
            check2 = checkTrajectorySimple(b, speed, root, dest, tangentOfLaunchAngle1, w);
        }
        return check1 || check2;
    }

    private static boolean isAboveBuilding(Building b, double speed, double tangentOfLaunchAngle, Point intersection, int h0) {
        double xDist = b.x - intersection.x;
        double yDist = b.y - intersection.y;
        double dist = Math.sqrt(xDist * xDist + yDist * yDist);
        double computedHeight = computeHeightForDistAndTan(h0, speed, tangentOfLaunchAngle, dist);
        return computedHeight >= b.z - 1e-9;
    }

    private static boolean checkTrajectorySimple(Building[][] b, double speed, Building root, Building dest, double tan, double w) {
        Segment s = new Segment(root.x, root.y, dest.x, dest.y);
        int h0 = root.z;
        for (int i = 0; i < b.length ; i ++) {
            for (int j = 0 ; j < b[i].length ; j++) {
                double x = b[i][j].x;
                double y = b[i][j].y;
                List<Point> polygon = new ArrayList<>();
                polygon.add(new Point(x - w/2, y - w/2));
                polygon.add(new Point(x + w/2, y - w/2));
                polygon.add(new Point(x + w/2, y + w/2));
                polygon.add(new Point(x - w/2, y + w/2));

                Point intersect = null;
                for (int k = 0 ; k < 4 ; k ++) {
                    Point p1 = polygon.get(k);
                    Point p2 = polygon.get((k+1)%polygon.size());
                    Segment polySeg = new Segment(p1.x, p1.y, p2.x, p2.y);
                    intersect = s.intersect(polySeg);
                    boolean inter = Line2D.linesIntersect(p1.x, p1.y, p2.x, p2.y, s.x1, s.y1, s.x2, s.y2);
                    if (intersect != null && intersect != Segment.OVERLAP) {
                        if (!inter) {
                            throw null;
                        }
                        if (!isAboveBuilding(b[i][j], speed, tan, intersect, h0)) {
                            return false;
                        }
                    } else {
                        if (inter) {
//                            System.out.println(s + " " + polySeg + " "  + intersect);
                            throw null;
                        }
                    }
                }
            }
        }

        return true;
    }

//    private static boolean checkTrajectory(Building[][] b, double speed, Building root, Building dest, double tan, Segment[] verticals, Segment[] horizontals, double w) {
//
//        Segment s = new Segment(root.x, root.y, dest.x, dest.y);
//        Map<Segment, Point> segmentPointMap = new HashMap<>();
//        int h0 = root.z;
//        for (int i = 0; i < horizontals.length; i++) {
//            Segment segment = horizontals[i];
//            Point intersect = s.intersect(segment);
//            if (intersect != null && intersect != Segment.OVERLAP) {
////                System.out.println(intersect + " = " + ("[" + root.i + ", " + root.j + "]") + " " + ("[" + dest.i + ", " + dest.j + "]") + " ----> " + segment);
//                segmentPointMap.put(segment, intersect);
//                if (verticals.length > 0) {
//                    if (intersect.x >= verticals[0].x1 - w && intersect.x <= verticals[0].x1) {
////                        System.out.println(("-> [" + i + ", " + 0 + "]") + " " + ("[" + (i + 1) + ", " + 0 + "]"));
//                        if (!isAboveBuilding(b[i][0], speed, tan, intersect, h0)
//                                || !isAboveBuilding(b[i + 1][0], speed, tan, intersect, h0)) {
//                            return false;
//                        }
//                    }
//                    for (int j = 0; j < verticals.length; j++) {
//                        if (intersect.x >= verticals[j].x1 && intersect.x <= verticals[j].x1 + w) {
////                            System.out.println(("-> [" + i + ", " + (j + 1) + "]") + " " + ("[" + (i + 1) + ", " + (j + 1) + "]"));
//                            if (!isAboveBuilding(b[i][j + 1], speed, tan, intersect, h0)
//                                    || !isAboveBuilding(b[i + 1][j + 1], speed, tan, intersect, h0)) {
//                                return false;
//                            }
//                        }
//                    }
//                } else {
////                    System.out.println(("-> [" + i + ", " + 0 + "]") + " " + ("[" + (i + 1) + ", " + 0 + "]"));
//                    if (!isAboveBuilding(b[i][0], speed, tan, intersect, h0)
//                            || !isAboveBuilding(b[i + 1][0], speed, tan, intersect, h0)) {
//                        return false;
//                    }
//                }
//            }
//        }
//
//        for (int i = 0; i < verticals.length; i++) {
//            Segment segment = verticals[i];
//            Point intersect = s.intersect(segment);
//            if (intersect != null && intersect != Segment.OVERLAP) {
////                System.out.println(intersect + " = " + ("[" + root.i + ", " + root.j + "]") + " " + ("[" + dest.i + ", " + dest.j + "]") + " ----> " + segment);
//                segmentPointMap.put(segment, intersect);
//                if (horizontals.length > 0) {
//                    if (intersect.y >= horizontals[0].y1 - w && intersect.y <= horizontals[0].y1) {
////                        System.out.println(("-< [" + 0 + ", " + i + "]") + " " + ("[" + 0 + ", " + (i + 1) + "]"));
//                        if (!isAboveBuilding(b[0][i], speed, tan, intersect, h0)
//                                || !isAboveBuilding(b[0][i + 1], speed, tan, intersect, h0)) {
//                            return false;
//                        }
//                    }
//                    for (int j = 0; j < horizontals.length; j++) {
//                        if (intersect.y >= horizontals[j].y1 && intersect.y <= horizontals[j].y1 + w) {
////                            System.out.println(("-< [" + (j + 1) + ", " + i + "]") + " " + ("[" + (j + 1) + ", " + (i + 1) + "]"));
//                            if (!isAboveBuilding(b[j + 1][i], speed, tan, intersect, h0)
//                                    || !isAboveBuilding(b[j + 1][i + 1], speed, tan, intersect, h0)) {
//                                return false;
//                            }
//                        }
//                    }
//                } else {
////                    System.out.println(("-< [" + 0 + ", " + i + "]") + " " + ("[" + 0 + ", " + (i + 1) + "]"));
//                    if (!isAboveBuilding(b[0][i], speed, tan, intersect, h0)
//                            || !isAboveBuilding(b[0][i + 1], speed, tan, intersect, h0)) {
//                        return false;
//                    }
//                }
//            }
//        }
//
////        System.out.println("end for " + ("[" + root.i + ", " + root.j + "]") + " " + ("[" + dest.i + ", " + dest.j + "]"));
////        System.out.println();
//        return true;
//    }

    private static double getHorizontalDistBetweenSquared(Building b1, Building b2) {
        return (b1.x - b2.x) * (b1.x - b2.x) + (b1.y - b2.y) * (b1.y - b2.y);
    }

    private static double getTangentOfLaunchAngle(double speed, double dist, double h, int coef) {
        double a = G * dist * dist / (2 * speed * speed);
        double b = -dist;
        double c = a + h;

        double delta = b * b - 4 * a * c;

        if (delta < 0) {
            return -1;
        }

        return (-b + coef * Math.sqrt(delta)) / (2 * a);
    }

    private static double getAngle(double v, double dist, double h) {
        double start = Math.PI / 2;
        double end = Math.PI / 4;
        double computedY = Double.MAX_VALUE;
        double x = 0;

        while (Math.abs(computedY - h) > 0.000005) {
            x = (start + end) / 2;
            if (start == x || end == x) {
                return -1;
            }

            double vy = v * Math.sin(x);
            double timeToHighest = vy / G;
            double vx = v * Math.cos(x);
            double timeToDist = dist / vx;

            if (timeToDist < timeToHighest - 1e-9) {
                end = x;
                continue;
            }

            computedY = computeHeightForDistAndTan(0, v, Math.tan(x), dist);
            if (start - end < 0.00000000000000000000001) {
                return -1;
            }
            if (computedY < h) {
                start = x;
            } else {
                end = x;
            }
        }
        return x;
    }

    private static double computeHeightForDistAndTan(int h0, double speed, double tan, double dist) {
        return h0 + dist * tan - G * dist * dist * (1 + tan * tan) / (2 * speed * speed);
    }

    public static int[] readArrayLine(String line, int n) {
        return readArrayLine(line, n, null, 0);
    }

    private static int[] readArrayLine(String line, int n, int array[], int pos) {
        int[] ret = array == null ? new int[n] : array;
        int start = 0;
        int end = line.indexOf(' ', start);
        for (int i = pos; i < pos + n; i++) {
            if (end > 0)
                ret[i] = Integer.parseInt(line.substring(start, end));
            else
                ret[i] = Integer.parseInt(line.substring(start));
            start = end + 1;
            end = line.indexOf(' ', start);
        }
        return ret;
    }
}
