# This Source Code Form is subject to the terms of the Mozilla Public
# License, v. 2.0. If a copy of the MPL was not distributed with this
# file, You can obtain one at http://mozilla.org/MPL/2.0/.

# Generates tables of background images which correspond with border images for
# creating reftests. Input is the filename containing input defined below (a subset
# of the allowed CSS border properties). An html representation of a table is
# output to stdout.
#
# Usage: python gen-refs.py input_filename
#
# Input must take the form (order is not important, nothing is optional, distance in order top, right, bottom, left):
# width: p;
# height: p;
# border-width: p;
# border-image-source: ...;
# border-image-slice: p p p p;
# note that actually border-image-slice takes numbers without px, which represent pixels anyway (or at least coords)
# border-image-width: np np np np;
# border-image-repeat: stretch | repeat | round;
# border-image-outset: np np np np;
#
# where:
# p ::= n'px'
# np ::= n | p
#
# Assumes there is no intrinsic size for the border-image-source, so uses
# the size of the border image area.

import sys

class Point:
  def __init__(self, w=0, h=0):
    self.x = w
    self.y = h
class Size:
  def __init__(self, w=0, h=0):
    self.width = w
    self.height = h
class Rect:
  def __init__(self, x=0, y=0, x2=0, y2=0):
    self.x = x
    self.y = y
    self.x2 = x2
    self.y2 = y2
  def width(self):
    return self.x2 - self.x
  def height(self):
    return self.y2 - self.y

class Props:
  def __init__(self):
    self.size = Size()

class np:
  def __init__(self, n, p):
    self.n = n
    self.p = p

  def get_absolute(self, ref):
    if not self.p == 0:
      return self.p
    return self.n * ref

def parse_p(tok):
  if tok[-2:] == "px":
    return float(tok[:-2])
  print "Whoops, not a pixel value " + tok

def parse_np(tok):
  if tok[-2:] == "px":
    return np(0, float(tok[:-2]))
  return np(float(tok), 0)

def parse(filename):
  f = open(filename, "r")
  props = Props()
  for l in f:
    l = l.strip()
    if not l[-1] == ";":
      continue
    toks = l[:-1].split()
    if toks[0] == "border-width:":
      props.width = parse_p(toks[1])
    if toks[0] == "height:":
      props.size.height = parse_p(toks[1])
    if toks[0] == "width:":
      props.size.width = parse_p(toks[1])
    if toks[0] == "border-image-source:":
      props.source = l[l.find(":")+1:l.rfind(";")].strip()
    if toks[0] == "border-image-repeat:":
      props.repeat = toks[1]
    if toks[0] == "border-image-slice:":
      props.slice = map(parse_p, toks[1:5])
    if toks[0] == "border-image-width:":
      props.image_width = map(parse_np, toks[1:5])
    if toks[0] == "border-image-outset:":
      props.outset = map(parse_np, toks[1:5])
  f.close()
  return props

# the result of normalisation is that all sizes are in pixels and the size,
# widths, and outset have been normalised to a size and width - the former is
# the element's interior, the latter is the width of the drawn border.
def normalise(props):
  result = Props()
  result.source = props.source
  result.repeat = props.repeat
  result.width = map(lambda x: x.get_absolute(props.width), props.image_width)
  outsets = map(lambda x: x.get_absolute(props.width), props.outset)
  result.size.width = props.size.width + 2*props.width + outsets[1] + outsets[3]
  result.size.height = props.size.height + 2*props.width + outsets[0] + outsets[2]
  result.slice = props.slice
  for i in [0,2]:
    if result.slice[i] > result.size.height:
      result.slice[i] = result.size.height
    if result.slice[i+1] > result.size.width:
      result.slice[i+1] = result.size.width

  return result

def check_parse(props):
  if not hasattr(props, 'source'):
    print "missing border-image-source"
    return False
  if not hasattr(props.size, 'width'):
    print "missing width"
    return False
  if not hasattr(props.size, 'height'):
    print "missing height"
    return False
  if not hasattr(props, 'width'):
    print "missing border-width"
    return False
  if not hasattr(props, 'image_width'):
    print "missing border-image-width"
    return False
  if not hasattr(props, 'slice'):
    print "missing border-image-slice"
    return False
  if not hasattr(props, 'repeat') or (props.repeat not in ["stretch", "repeat", "round"]):
    print "missing or incorrect border-image-repeat '" + props.repeat + "'"
    return False
  if not hasattr(props, 'outset'):
    print "missing border-image-outset"
    return False

  return True

def check_normalise(props):
  if not hasattr(props, 'source'):
    print "missing border-image-source"
    return False
  if not hasattr(props.size, 'width'):
    print "missing width"
    return False
  if not hasattr(props.size, 'height'):
    print "missing height"
    return False
  if not hasattr(props, 'slice'):
    print "missing border-image-slice"
    return False
  if not hasattr(props, 'repeat') or (props.repeat not in ["stretch", "repeat", "round"]):
    print "missing or incorrect border-image-repeat '" + props.repeat + "'"
    return False

  return True

class Tile:
  def __init__(self):
    self.slice = Rect()
    self.border_width = Rect()

# throughout, we will use arrays for nine-patches, the indices correspond thusly:
# 0 1 2
# 3 4 5
# 6 7 8

# Compute the source tiles' slice and border-width sizes
def make_src_tiles():
  tiles = [Tile() for i in range(9)]

  rows = [range(3*i, 3*(i+1)) for i in range(3)]
  cols = [[i, i+3, i+6] for i in range(3)]

  row_limits_slice = [0, props.slice[3], props.size.width - props.slice[1], props.size.width]
  row_limits_width = [0, props.width[3], props.size.width - props.width[1], props.size.width]
  for r in range(3):
    for t in [tiles[i] for i in cols[r]]:
      t.slice.x = row_limits_slice[r]
      t.slice.x2 = row_limits_slice[r+1]
      t.border_width.x = row_limits_width[r]
      t.border_width.x2 = row_limits_width[r+1]

  col_limits_slice = [0, props.slice[0], props.size.height - props.slice[2], props.size.height]
  col_limits_width = [0, props.width[0], props.size.height - props.width[2], props.size.height]
  for c in range(3):
    for t in [tiles[i] for i in rows[c]]:
      t.slice.y = col_limits_slice[c]
      t.slice.y2 = col_limits_slice[c+1]
      t.border_width.y = col_limits_width[c]
      t.border_width.y2 = col_limits_width[c+1]

  return tiles

def compute(props):
  tiles = make_src_tiles()

  # corners scale easy
  for t in [tiles[i] for i in [0, 2, 6, 8]]:
    t.scale = Point(t.border_width.width()/t.slice.width(), t.border_width.height()/t.slice.height())
  # edges are by their secondary dimension
  for t in [tiles[i] for i in [1, 7]]:
    t.scale = Point(t.border_width.height()/t.slice.height(), t.border_width.height()/t.slice.height())
  for t in [tiles[i] for i in [3, 5]]:
    t.scale = Point(t.border_width.width()/t.slice.width(), t.border_width.width()/t.slice.width())
  # the middle is scaled by the factors for the top and left edges
  tiles[4].scale = Point(tiles[1].scale.x, tiles[3].scale.y)

  # the size of a source tile for the middle section
  src_tile_size = Size(tiles[4].slice.width()*tiles[4].scale.x, tiles[4].slice.height()*tiles[4].scale.y)

  # the size of a single destination tile in the central part
  dest_tile_size = Size()
  if props.repeat == "stretch":
    dest_tile_size.width = tiles[4].border_width.width()
    dest_tile_size.height = tiles[4].border_width.height()
    for t in [tiles[i] for i in [1, 7]]:
      t.scale.x = t.border_width.width()/t.slice.width()
    for t in [tiles[i] for i in [3, 5]]:
      t.scale.y = t.border_width.height()/t.slice.height()
  elif props.repeat == "repeat":
    dest_tile_size = src_tile_size
  elif props.repeat == "round":
    dest_tile_size.width = tiles[4].border_width.width() / math.ceil(tiles[4].border_width.width() / src_tile_size.width)
    dest_tile_size.height = tiles[4].border_width.height() / math.ceil(tiles[4].border_width.height() / src_tile_size.height)
    for t in [tiles[i] for i in [1, 4, 7]]:
      t.scale.x = dest_tile_size.width/t.slice.width()
    for t in [tiles[i] for i in [3, 4, 5]]:
      t.scale.y = dest_tile_size.height/t.slice.height()
  else:
    print "Whoops, invalid border-image-repeat value"

  # catch overlapping slices. Its easier to deal with it here than to catch
  # earlier and have to avoid all the divide by zeroes above
  for t in tiles:
    if t.slice.width() < 0:
      t.scale.x = 0
    if t.slice.height() < 0:
      t.scale.y = 0

  tiles_h = int(math.ceil(tiles[4].border_width.width()/dest_tile_size.width)+2)
  tiles_v = int(math.ceil(tiles[4].border_width.height()/dest_tile_size.height)+2)

  # if border-image-repeat: repeat, then we will later center the tiles, that
  # means we need an extra tile for the two 'half' tiles at either end
  if props.repeat == "repeat":
    if tiles_h % 2 == 0:
      tiles_h += 1
    if tiles_v % 2 == 0:
      tiles_v += 1
  dest_tiles = [Tile() for i in range(tiles_h * tiles_v)]

  # corners
  corners = [(0, 0), (tiles_h-1, 2), (tiles_v*(tiles_h-1), 6), (tiles_v*tiles_h-1, 8)]
  for d,s in corners:
    dest_tiles[d].size = Size(tiles[s].scale.x*props.size.width, tiles[s].scale.y*props.size.height)
    dest_tiles[d].dest_size = Size(tiles[s].border_width.width(), tiles[s].border_width.height())
  dest_tiles[0].offset                   = Point(0, 0)
  dest_tiles[tiles_h-1].offset           = Point(tiles[2].border_width.width() - dest_tiles[tiles_h-1].size.width, 0)
  dest_tiles[tiles_v*(tiles_h-1)].offset = Point(0, tiles[6].border_width.height() - dest_tiles[tiles_v*(tiles_h-1)].size.height)
  dest_tiles[tiles_v*tiles_h-1].offset   = Point(tiles[8].border_width.width() - dest_tiles[tiles_h*tiles_v-1].size.width, tiles[8].border_width.height() - dest_tiles[tiles_h*tiles_v-1].size.height)

  # horizontal edges
  for i in range(1, tiles_h-1):
    dest_tiles[i].size                       = Size(tiles[1].scale.x*props.size.width, tiles[1].scale.y*props.size.height)
    dest_tiles[(tiles_v-1)*tiles_h + i].size = Size(tiles[7].scale.x*props.size.width, tiles[7].scale.y*props.size.height)
    dest_tiles[i].dest_size                       = Size(dest_tile_size.width, tiles[1].border_width.height())
    dest_tiles[(tiles_v-1)*tiles_h + i].dest_size = Size(dest_tile_size.width, tiles[7].border_width.height())
    dest_tiles[i].offset                       = Point(-tiles[1].scale.x*tiles[1].slice.x, -tiles[1].scale.y*tiles[1].slice.y)
    dest_tiles[(tiles_v-1)*tiles_h + i].offset = Point(-tiles[7].scale.x*tiles[7].slice.x, -tiles[7].scale.y*tiles[7].slice.y)

  # vertical edges
  for i in range(1, tiles_v-1):
    dest_tiles[i*tiles_h].size       = Size(tiles[3].scale.x*props.size.width, tiles[3].scale.y*props.size.height)
    dest_tiles[(i+1)*tiles_h-1].size = Size(tiles[5].scale.x*props.size.width, tiles[5].scale.y*props.size.height)
    dest_tiles[i*tiles_h].dest_size       = Size(tiles[3].border_width.width(), dest_tile_size.height)
    dest_tiles[(i+1)*tiles_h-1].dest_size = Size(tiles[5].border_width.width(), dest_tile_size.height)
    dest_tiles[i*tiles_h].offset       = Point(-tiles[3].scale.x*tiles[3].slice.x, -tiles[3].scale.y*tiles[3].slice.y)
    dest_tiles[(i+1)*tiles_h-1].offset = Point(-tiles[5].scale.x*tiles[5].slice.x, -tiles[5].scale.y*tiles[5].slice.y)

  # middle
  for i in range(1, tiles_v-1):
    for j in range(1, tiles_h-1):
      dest_tiles[i*tiles_h+j].size = Size(tiles[4].scale.x*props.size.width, tiles[4].scale.y*props.size.height)
      dest_tiles[i*tiles_h+j].offset = Point(-tiles[4].scale.x*tiles[4].slice.x, -tiles[4].scale.y*tiles[4].slice.y)
      dest_tiles[i*tiles_h+j].dest_size = dest_tile_size

  # edge and middle tiles are centered with border-image-repeat: repeat
  # we need to change the offset to take account of this and change the dest_size
  # of the tiles at the sides of the edges if they are clipped
  if props.repeat == "repeat":
    diff_h = ((tiles_h-2)*dest_tile_size.width - tiles[4].border_width.width()) / 2
    diff_v = ((tiles_v-2)*dest_tile_size.height - tiles[4].border_width.height()) / 2
    for i in range(0, tiles_h):
      dest_tiles[tiles_h + i].dest_size.height -= diff_v
      dest_tiles[tiles_h + i].offset.y -= diff_v #* tiles[4].scale.y
      dest_tiles[(tiles_v-2)*tiles_h + i].dest_size.height -= diff_v
    for i in range(0, tiles_v):
      dest_tiles[i*tiles_h + 1].dest_size.width -= diff_h
      dest_tiles[i*tiles_h + 1].offset.x -= diff_h #* tiles[4].scale.x
      dest_tiles[(i+1)*tiles_h-2].dest_size.width -= diff_h

  # output the table to simulate the border
  print "<table>"
  for i in range(tiles_h):
    print "<col style=\"width: " + str(dest_tiles[i].dest_size.width) + "px;\">"
  for i in range(tiles_v):
    print "<tr style=\"height: " + str(dest_tiles[i*tiles_h].dest_size.height) + "px;\">"
    for j in range(tiles_h):
      width = dest_tiles[i*tiles_h+j].size.width
      height = dest_tiles[i*tiles_h+j].size.height
      # catch any tiles with negative widths/heights
      # this happends when the total of the border-image-slices > borde drawing area
      if width <= 0 or height <= 0:
        print "  <td style=\"background: white;\"></td>"
      else:
        print "  <td style=\"background-image: " + props.source + "; background-size: " + str(width) + "px " + str(height) + "px; background-position: " + str(dest_tiles[i*tiles_h+j].offset.x) + "px " + str(dest_tiles[i*tiles_h+j].offset.y) + "px;\"></td>"
    print "</tr>"
  print "</table>"


# start here
args = sys.argv[1:]
if len(args) == 0:
  print "whoops: no source file"
  exit(1)


props = parse(args[0])
if not check_parse(props):
  print dir(props)
  exit(1)
props = normalise(props)
if not check_normalise(props):
  exit(1)
compute(props)