Google Code Jam/Crop Triangles

From HaskellWiki

Problem

Some pranksters have watched too much Discovery Channel and now they want to build a crop triangle during the night. They want to build it inside a large crop that looks like an evenly spaced grid from above. There are some trees planted on the field. Each tree is situated on an intersection of two grid lines (a grid point). The pranksters want the vertices of their crop triangle to be located at these trees. Also, for their crop triangle to be more interesting they want the center of that triangle to be located at some grid point as well. We remind you that if a triangle has the vertices (x1, y1), (x2, y2) and (x3, y3), then the center for this triangle will have the coordinates ((x1 + x2 + x3) / 3, (y1 + y2 + y3) / 3).

You will be given a set of points with integer coordinates giving the location of all the trees on the grid. You are asked to compute how many triangles you can form with distinct vertexes in this set of points so that their center is a grid point as well (i.e. the center has integer coordinates).

If a triangle has area 0 we will still consider it a valid triangle.

Input

The first line of input gives the number of cases, N. N test cases follow. Each test case consists of one line containing the integers n, A, B, C, D, x0, y0 and M separated by exactly one space. n will be the number of trees in the input set. Using the numbers n, A, B, C, D, x0, y0 and M the following pseudocode will print the coordinates of the trees in the input set. mod indicates the remainder operation.

The parameters will be chosen such that the input set of trees will not have duplicates.

X = x0, Y = y0
print X, Y
for i = 1 to n-1
  X = (A * X + B) mod M
  Y = (C * Y + D) mod M
  print X, Y

Output

For each test case, output one line containing "Case #X: " where X is the test case number (starting from 1). This should be followed by an integer indicating the number of triangles which can be located at 3 distinct trees and has a center that is a grid point.

Limits

1 <= N <= 10,
0 <= A, B, C, D, x0, y0<= 109,
1 <= M <= 109.

Small dataset

3 <= n <= 100.

Large dataset

3 <= n <= 100000.

Sample

Input

2
4 10 7 1 2 0 1 20
6 2 0 2 1 1 2 11

Output

Case #1: 1
Case #2: 2

In the first test case, the 4 trees in the generated input set are (0, 1), (7, 3), (17, 5), (17, 7).

Solution

import Control.Applicative
import Data.List

test ((ax,ay),(bx,by),(cx,cy)) = (ax+bx+cx) `mod` 3 == 0 && 
	(ay+by+cy) `mod` 3 == 0

type Point = (Integer,Integer)
type Triangle = (Point,Point,Point)

triangles :: [Point] -> [Triangle]
triangles xs = do
	(a:as) <- tails (nub xs)
	(b:bs) <- tails as
	c <- bs
	return (a,b,c)

solution n a b c d x y m = length . filter test . triangles . nub $
	take (fromInteger n) (iterate (points a b c d m) (x,y)) where
	points a b c d m (x,y) = ((a * x + b) `mod` m ,(c * y + d) `mod` m)
	 
main = enumFromTo 1 <$> readLn >>= mapM_ doCase
  where doCase caseno = do
  		[n,a,b,c,d,x,y,m] <- map read . words <$> getLine
  		putStrLn $ "Case #" ++ show caseno ++ ": " ++ 
			show (solution n a b c d x y m)