R/curvial_split.R

Defines functions .verts.y .y_corrections .spline_corrections .curve_quality .curvi_split .curvi_split.fast .curvial_split

#' Find the curve that divides the polygon best (main function)
#'
#' @description function prepares the data to start the searching procedure of the curve that provides the best spatial separation.
#' As curve are generated using splines, matrix of knot coordinates are prepared. Also, polygon is rotated so that split line would be horizontal and at Y = 0, and start at X = 0.
#' @param poly.x a vector of x coordinates of a polygon perimeter points
#' @param poly.y a vector of y coordinates of a polygon perimeter points
#' @param min.x.id index of split line vertex in poly.x and poly.y objects that has lower x coordinate
#' @param max.x.id index of split line vertex in poly.x and poly.y objects that has lower y coordinate
#' @param b slope of a split line
#' @param samp.dat spatial subset of data
#' @param c.X.knots number of spline knots along the split line
#' @param c.Y.knots number of spline knots orthogonal to the split line
#' @param N.crit minimum number of observations required to establish
#' subdivision of a polygon.
#' @param N.rel.crit number from 0 to 0.5. The minimum allowed observation
#' proportion in polygon containing less data.
#' @param N.loc.crit Subdivision stopping criteria - number of unique locations.
#' Only meaningful when there are observations from the same location. Default
#' 0.2.
#' @param N.loc.rel.crit number from 0 to 0.5. The minimum allowed unique
#' locations proportions in polygon containing less locations.
#' @param S.cond minimum area required to establish subdivision of a plot
#' @param S.rel.crit number from 0 to 0.5. The minimum allowed area
#' proportion of a smaller polygon. Default 0.2.
#' @param S_org area of the polygon being divided.
#' @param c.max.iter.no maximum number of allowed iterations through the spline
#' knots.
#' @param c.fast.optim Logical (default TRUE). Determines when spline knots
#' are selected: TRUE - when first better curve than before is obtained, FALSE -
#' when the best curve is obtained (after all spline knots are checked).
#' @param trace.level Sting indicating the trace level. Can be one of the
#' following: "best", "main", "all"
#' @param trace.object String that informs what graphical information to show.
#' Can be either "curves", "straight" or "both".
#' @param c.corr.term term that defines how much the a problematic spline
#' will be corrected (in terms of proportion of polygon width where spline
#' intersects the polygon boundary) if the spline is not contained within the
#' plot. Small values recommended (default is 0.05).
#' @param pnts.col Color of data points, default is 1. Argument is used when
#' \code{trace} > 0. If set to NULL, data points will not be displayed.
#' @param straight.qual Quality of the best straight split-line.
#' @return A list of two elements: 1) curve in shape of a spline that produces
#' the best data separation; 2) quality of the division
#' @noRd
.curvial_split <- function(poly.x,poly.y,min.x.id,max.x.id,b,
                           samp.dat,c.X.knots=20,c.Y.knots=20,
                           N.crit,
                           N.rel.crit,N.loc.crit,N.loc.rel.crit,S.cond,
                           S.rel.crit, S_org, c.max.iter.no, c.fast.optim,
                           c.corr.term,
                           trace.object = trace.object,
                           trace.level = trace.level,
                           pnts.col = pnts.col,
                           straight.qual){
  #nustatau, kuris padalinimo linijos id yra kairej, kuris desinej
  # length of a split line
  #AE<-sqrt(sum((c(poly.x[min.x.id],
  #                poly.y[min.x.id])-c(poly.x[max.x.id],poly.y[max.x.id]))^2))
  #randu kampa, kuriuo reikia paversti poligona, kad padalinimo linija butu
  # horizontali
  teta<-atan(b)
  #paverciu poligona ir duomenis
  rot.pol.cords <- DescTools::Rotate(x=poly.x,y=poly.y,mx=poly.x[min.x.id],
                          my=poly.y[min.x.id],theta=-teta)
  rot.dat.cords <- DescTools::Rotate(x=samp.xy$x,y=samp.xy$y,mx=poly.x[min.x.id],
                          my=poly.y[min.x.id],theta=-teta)

  #pastumiu duomenis ir poligona, kad padalinimo linijos kairinis taskas butu
  # koordinaciu sistemos pradzioje
  rot.dat.cords <- data.frame(x = rot.dat.cords$x-poly.x[min.x.id],
                              y = rot.dat.cords$y-poly.y[min.x.id])
  #rot.data<-data.frame(data[,1],x = rot.dat.cords$x-poly.x[min.x.id],
   #                    y = rot.dat.cords$y-poly.y[min.x.id])
  #sukuriu duomenu masyva pasukto poligono
  rot.poli<-data.frame(x= rot.pol.cords$x-poly.x[min.x.id],
                       y= rot.pol.cords$y-poly.y[min.x.id])
  #padalinimo linijos pabaigoje y turetu buti 0, bet del paklaidu jis gali buti
  # nelygus 0 ir filtracija gali blogai ivykti - nunulinu
  rot.poli[-min.x.id,2] <- rot.poli[-min.x.id,2] - rot.poli[max.x.id,2] # removing error at the
  # end of the split line.
  AE <- rot.poli[max.x.id,1]
  #"atidarau" poligona
  # rot.poli<-rot.poli[-nrow(rot.poli),]
  #nufiltruoju virsutine ir apatine poligono dali, t
  #askai tampa organizuoti palei poligono kreive is kaires i desine
  rot.poli.up<-.close_poly(.split_poly(
    polygon = rot.poli,
    min_id = 1,
    split_ids = c(min.x.id,max.x.id),
    trivial_side = TRUE,
    poli_side = TRUE))
  poli.up <- .close_poly(.split_poly(
    polygon = data.frame(x = poly.x, y = poly.y),
    min_id = 1,
    split_ids = c(min.x.id,max.x.id),
    trivial_side = TRUE,
    poli_side = TRUE))

  rot.poli.do<-.close_poly(.split_poly(
    polygon = rot.poli,
    min_id = 1,
    split_ids = c(min.x.id,max.x.id),
    trivial_side = TRUE,
    poli_side = FALSE))
  poli.do <- .close_poly(.split_poly(
    polygon = data.frame(x = poly.x, y = poly.y),
    min_id = 1,
    split_ids = c(min.x.id,max.x.id),
    trivial_side = TRUE,
    poli_side = FALSE))

  #randu vidinio pataisyto poligono apatine ir virsutine dali
  bottom.inner.poli <-
    .inner_poly_hull(polygon = rot.poli.do[!duplicated(rot.poli.do),])
  upper.inner.poli <-
    .inner_poly_hull(polygon = rot.poli.up[!duplicated(rot.poli.up),])
  if( bottom.inner.poli[[2]] == 1 ){

    change <- bottom.inner.poli
    bottom.inner.poli <- upper.inner.poli
    upper.inner.poli <- change

    change <- rot.poli.do
    rot.poli.do <- rot.poli.up
    rot.poli.up <- change

    change <- poli.up
    poli.up <- poli.do
    poli.do <- change

  }

  # The following steps are needed to identify points that lie exactly
  # on the contour of polygon, because after rotation small errors are
  # introduced and because of them these points might be detected as lying
  # oustide polygon. For this reason, some points might be not considered
  # when searching for the best curvial split line. Thus, the ids of these
  # points are found, and added, if missed, to remove this bug.
  #
  {
  # unr_ids: filter point ids by unrotated  up & bottom poly
  unr_ids_up <- .get_ids(polygon = poli.up, xy_dat = samp.xy)
  unr_ids_do <- .get_ids(polygon = poli.do, xy_dat = samp.xy)
  # r_ids: filter point ids by rotated up & bottom poly
  r_ids_up <- .get_ids(polygon = rot.poli.up, xy_dat = rot.dat.cords)
  r_ids_do <- .get_ids(polygon = rot.poli.do, xy_dat = rot.dat.cords)

  # m_ids: find unr_ids that are missing in r_ids
  m_ids_up <- unr_ids_up[which(!unr_ids_up %in% r_ids_up)]
  m_ids_do <- unr_ids_do[which(!unr_ids_do %in% r_ids_do)]
  # theoretically m_ids are points on polygon perimeter

  # if m_ids is not empty  (filter.all?):
  # split_pt_ids find ids of data points that are on a straight split line
  # - point in poly -  3, when up or bottom, but 1 when whole.

  if (length(m_ids_up) > 0 | length(m_ids_do) > 0 ){
    ids.on.pol.updo <-
    unique(c(which(.point.in.polygon(pol.x = poli.up[,1],
                               pol.y = poli.up[,2],
                               point.x = samp.xy$x,
                               point.y = samp.xy$y) == 3),
    which(.point.in.polygon(pol.x = poli.do[,1],
                               pol.y = poli.do[,2],
                               point.x = samp.xy$x,
                               point.y = samp.xy$y) == 3)))
    split_pt_ids <- ids.on.pol.updo[
      ids.on.pol.updo %in%
        which(.point.in.polygon(pol.x = poly.x,
                                   pol.y = poly.y,
                                   point.x = samp.xy$x,
                                   point.y = samp.xy$y) == 1)]
    # theoretically split_pt_ids are points on straight split_line, but not
    # on polygon perimeter.
    # This solution should be added to .get_ids, when filter.all = FALSE
    #
    # if split_pt_ids is not empty:
    #  remove split_pt_ids  from m_ids
    if (length(split_pt_ids) > 0){
      if (any(split_pt_ids %in% m_ids_up))
        m_ids_up <- m_ids_up[-which(split_pt_ids == m_ids_up)]
      if (any(split_pt_ids %in% m_ids_do))
        m_ids_do <- m_ids_do[-which(split_pt_ids == m_ids_do)]}

  }

  # if m_ids not empty:
  #  inside curvi_split & curvi_split_fast, when using .get_ids in
  #  .curve_quality, additionally add m_ids to the output of ids.
  m_ids <- list(m_ids_up, m_ids_do)
  }
  rot.poli.do <- rot.poli.do[-nrow(rot.poli.do),]
  rot.poli.up <- rot.poli.up[-nrow(rot.poli.up),]
  #pasirupinam splino knot'ais pradiniais

  split.line.x <- seq(0,AE,length.out = c.X.knots)
  split.line.x[c.X.knots] <- AE
  #ant pataisyto poligono reikia rasti Y koordinates duotom x koordinatem,
  # atitinkanciom padalinimo linijos atkarpu galus
  up.y <- numeric(c.X.knots)
  do.y <- numeric(c.X.knots)
  for(i in 2:(length(split.line.x)-1)){
    up.y[i]<-.y.online(x=upper.inner.poli[[1]]$x,y=upper.inner.poli[[1]]$y,
                       x3=split.line.x[i])
    do.y[i]<-.y.online(x=bottom.inner.poli[[1]]$x,y=bottom.inner.poli[[1]]$y,
                       x3=split.line.x[i])
  }
  #randam ploti pataisyto poligono
  range <- up.y-do.y
  #nustatom i kiek daliu dalinsim polygona pagal Y koordinate
  B <- seq(0,1,length.out = c.Y.knots)
  #sugeneruojam matrica su testuojamu spline knotu y koordinatem.
  #Kiekvienas stulpelis atitinka skirtinga X verte ant padalinimo linijos
  #spit.line.x, kiekviena eilute atitinka skirtinga poligono pjuvio ties x
  # koordinate dali - virsutines eilutes apatine poligono dalis
  knot.y.matrix <- matrix(B, c.Y.knots, 1) %*% matrix(range, 1, c.X.knots) +
    matrix(rep(do.y, each = c.Y.knots), c.Y.knots, c.X.knots)
  # if (testid == 17){
  #   trace.object <- "curve"
  #   trace.level <- "main"
  # }
  .visualise_splits.curve_start(what = trace.object,
                                rot.dat.cords, pnts.col, rot.poli, AE, c.X.knots,
                                split.line.x, c.Y.knots, knot.y.matrix)

  #randam geriausia padalinimo kreive
  if (c.fast.optim){
  environment(.curvi_split.fast) <- environment()

  best_curvi_split <- .curvi_split.fast(
    knot.y.matrix,split.line.x,Xup=upper.inner.poli[[1]]$x,
    Xdown=bottom.inner.poli[[1]]$x,Yup=upper.inner.poli[[1]]$y,
    Ydown=bottom.inner.poli[[1]]$y,
    N.crit,
    N.rel.crit,N.loc.crit,N.loc.rel.crit,S.cond,
    S.rel.crit,S_org,c.Y.knots,c.X.knots,rot.poli.up,rot.poli.do,
    rot.dat.cords,c.max.iter.no = c.max.iter.no, c.corr.term = c.corr.term,
    trace.object = trace.object, trace.level = trace.level,
    straight.qual = straight.qual,samp.dat = samp.dat,
    m_ids = m_ids
    )
  } else {
    environment(.curvi_split) <- environment()
    best_curvi_split <- .curvi_split(
      knot.y.matrix,split.line.x,Xup=upper.inner.poli[[1]]$x,
      Xdown=bottom.inner.poli[[1]]$x,Yup=upper.inner.poli[[1]]$y,
      Ydown=bottom.inner.poli[[1]]$y,
      N.crit,
      N.rel.crit,N.loc.crit,N.loc.rel.crit,S.cond,
      S.rel.crit,S_org,c.Y.knots,c.X.knots,rot.poli.up,rot.poli.do,
      rot.dat.cords,c.max.iter.no = c.max.iter.no, c.corr.term = c.corr.term,
      trace.object = trace.object, trace.level = trace.level,
      straight.qual = straight.qual,samp.dat = samp.dat,
      m_ids = m_ids
    )
  }

# Up --> Yup, Down --> Ydown; xp --> x; yp --> y.  visur pakeist
  best.curve <- data.frame(x=best_curvi_split[[1]]$x+poly.x[min.x.id],
                           y=best_curvi_split[[1]]$y+poly.y[min.x.id])
  best.curve <- DescTools::Rotate(x=best.curve$x,y=best.curve$y,
                                  mx=poly.x[min.x.id], my=poly.y[min.x.id],
                                  theta=teta)
  return(list(best.curve, best_curvi_split[[2]]))
}

#' Find the curve that divides the polygon best (recursive function)
#'
#' @description At this stage of an algorithm, polygon is rotated so that the split line is horizontal and positioned along X axis (at Y = 0), and split line starts at X = 0.
#' Function iterates trough knot matrix several times (maximum allowed number of times is defined by c.max.iter.no argument) and tries different splines to seperate data in space.
#' Iteration through knots starts from left to right along the split line. For each position along the split line several positions orthogonal to the split line are tried.
#' The best orthogonal positions are estimated from a spline model (Split Quality ~ Y coordinate of a knot for a given X coordinate)
#' So the best Y knot coordinate is provided for each knot along X. Thus, knots along the split line are fixated, but along Y are not. When one iteration is complete, the next
#' one starts in different direction (e.g. from right to left along the split line).
#'
#' @param knot.y.matrix a matrix with Y coordinates in columns for each knot along the split line
#' @param split.line.x a vector of x coordinates for each knot along the split line
#' @param Xup X coordinates of upper part (above split line) of standartized polygon
#' @param Xdown X coordinates of lower part (below split line) of standartized polygon
#' @param Yup Y coordinates of upper part (above split line) of standartized polygon
#' @param Ydown Y coordinates of lower part (below split line) of standartized polygon
#' @param N.crit minimum number of observations required to establish
#' subdivision of a polygon.
#' @param N.rel.crit number from 0 to 0.5. The minimum allowed observation
#' proportion in polygon containing less data.
#' @param N.loc.crit Subdivision stopping criteria - number of unique locations.
#' Only meaningful when there are observations from the same location. Default
#' 0.2.
#' @param N.loc.rel.crit number from 0 to 0.5. The minimum allowed unique
#' locations proportions in polygon containing less locations.
#' @param S.cond minimum area required to establish subdivision of a plot
#' @param S.rel.crit number from 0 to 0.5. The minimum allowed area
#' proportion of a smaller polygon. Default 0.2.
#' @param S_org area of the polygon being divided.
#' @param c.Y.knots number of spline knots orthogonal to the split line
#' @param c.X.knots number of spline knots along the split line
#' @param rot.poli.up rotated, but not standartized, open upper polygon
#' @param rot.poli.do rotated, but not standartized, open lower polygon
#' @param rot.dat.cords rotated coordinates of data
#' @param c.max.iter.no allowed number of iterations trough columns of spline
#' knots
#' @param c.corr.term term that defines how much the a problematic spline
#' will be corrected (in terms of proportion of polygon width where spline
#' intersects the polygon boundary) if the spline is not contained within the
#' plot. Small values recommended (default is 0.05).
#' @param trace.level Sting indicating the trace level. Can be one of the
#' following: "best", "main", "all"
#' @param trace.object String that informs what graphical information to show.
#' Can be either "curves", "straight" or "both".
#' @param straight.qual Quality of the best straight split-line.
#' @param samp.dat spatial subset of data
#' @param m_ids list of data point IDs from up and bottom polygon, respectively,
#' that lie on the perimeter of polygon, but are not correctly filtered with
#' .get_ids and therefore should be added manually to the output of .get_ids.
#' @return A list of two elements: 1) rotated (not suitable for the original polygon) curve in shape of a spline that produces the best data separation; 2) quality of the division
#' @importFrom stats spline
#' @author Liudas Daumantas
#' @noRd
.curvi_split.fast<-function(knot.y.matrix,split.line.x,Xup,Xdown,Yup,Ydown,
                            N.crit,N.rel.crit,N.loc.crit,N.loc.rel.crit,S.cond,
                            S.rel.crit,S_org,c.Y.knots,c.X.knots,rot.poli.up,
                            rot.poli.do,rot.dat.cords,c.max.iter.no = c.max.iter.no,
                            c.corr.term,trace.object = trace.object,
                            trace.level = trace.level, teta = teta, straight.qual,
                            samp.dat, m_ids){
  best.y.knots <- numeric(c.X.knots)
  y.cord.quality <- numeric(c.Y.knots-2)
  any.qual <- numeric()
  best.y.coords <- numeric(c.X.knots-2)
  best.qual <- straight.qual
  best.old.curve <- data.frame(x = numeric(0), y = numeric(0))
  best.y <- 0
  knot.y.matrix <- knot.y.matrix[-c(1,c.Y.knots),-c(1,c.X.knots)]
  knot.state <- !logical(c.X.knots-2)
  counter <- 0
  it <- 1
  SS <- list(NaN,NaN)

  while(it <= c.max.iter.no) {
    if (all(!knot.state)) break
    if(it%%2==1){
      if(it==1){
        seq.of.x.knots <- (1:(c.X.knots-2))[knot.state]
      } else {
        seq.of.x.knots <- (2:(c.X.knots-2))[knot.state[-1]]
      }} else {
        seq.of.x.knots <- ((c.X.knots-3):1)[rev(knot.state)[-1]]
      }
    for (l in seq.of.x.knots) {

      for (k in 1:(c.Y.knots-2)) {
        #isbandom vis kita Y koordinate knotui su duota x koordinate
        best.y.knots[1+l] <- knot.y.matrix[k,l]
        #sugeneruojam splina per knotus
        curve <- stats::spline(split.line.x,best.y.knots,n=100)
        #iteratyviai darom, kol nebemeta klaidos:
        #patikrinam ar poligono viduj kreive, gaunam pataisos tasku koordinates
        #skeliam split.line.x ir best.y.knots i tiek daliu, kiek reikia ir
        # reikiamose vietose pridedam koordinates
        #sukuriam nauja split.line.x best.y.knots varianta
        #kartojam splina
        curve <- .spline_corrections(curve,Xup,Xdown,Yup,Ydown,best.y.knots,
                                     split.line.x,c.corr.term=c.corr.term)
        counter <- counter + 1
        if (!is.null(curve)){
          .visualise_splits.try_curve(what = trace.object,level = trace.level,
                                      counter, curve)

          #ivertinam poligono padalinimo su sugeneruota kreive kokybe
          environment(.curve_quality) <- environment()
          SS <- .curve_quality(curve=curve,rot.poli.up, rot.poli.do,
                               rot.dat.cords,samp.dat,N.crit,
                               N.rel.crit,N.loc.crit,N.loc.rel.crit,S.cond,
                               S.rel.crit,S_org, m_ids)
        } else {
          SS[[1]] <- NaN
          SS[[2]] <- "Too complex geometry"
        }

        if ( SS[[2]] != "" ){
          y.cord.quality[k] <- NaN
          message <- SS[[2]]
          .visualise_splits.bad_curve(what = trace.object,level = trace.level,
                                      curve, message)
        } else{
          y.cord.quality[k] <- SS[[1]]
          any.qual <- c(any.qual,SS[[1]])
          if (.comp(SS[[1]],best.qual)){
            .visualise_splits.good_curve(what = trace.object,
                                         level = trace.level,
                                         curve, SS, best.old.curve)
            best.old.curve <- curve
            best.qual <- SS[[1]]
            .visualise_splits.selected_knot(what = trace.object,
                                            level = trace.level,
                                            x = split.line.x[l+1],
                                            y = knot.y.matrix[k,l],
                                            old.knot.y = best.y.coords[l])
            best.y.coords[l] <- knot.y.matrix[k,l]
            knot.state <- rep(TRUE, c.X.knots-2)
          } else {
            message <- paste0("Poor split quality.\n","Obtained: ",
                              round(SS[[1]],2),
                              "\nRequired: ",c.sign, round(best.qual,2))
            .visualise_splits.bad_curve(what = trace.object,level = trace.level,
                                        curve, message)
          }
        }
        #fiksuojam verte
      }
      knot.state[l] <- FALSE
      #sugeneruojam spline, kurio X asyje yra knotu Y koordinates, o Y asyje
      #padalinimo kreives kokybe
      #skirta interpoliuojant aproksimuoti tarpiniu Y koordinaciu vertes
      if (!(sum(!is.nan(y.cord.quality)) <= 3 |
            length(unique(y.cord.quality)) <= 2)){
        proj <- stats::spline(knot.y.matrix[,l], y.cord.quality, n=100)
        if (.comp(.minormax(proj$y), best.qual)){
          test.knots <- best.y.knots
          test.knots[1+l] <- proj$x[.which_minormax(proj$y)]
          curve.test <- stats::spline(split.line.x,test.knots,n=100)
          curve.test <- .spline_corrections(curve.test,Xup,Xdown,Yup,Ydown,test.knots,
                                            split.line.x,c.corr.term = c.corr.term)
          if (!is.null(curve.test)){
            .visualise_splits.try_int_curve(what = trace.object,
                                            level = trace.level,
                                            curve = curve.test,
                                            x = split.line.x[l+1],
                                            y = test.knots[1+l])
            environment(.curve_quality) <- environment()
            SS <- .curve_quality(curve.test, rot.poli.up, rot.poli.do,
                                 rot.dat.cords, samp.dat, N.crit,
                                 N.rel.crit,N.loc.crit,N.loc.rel.crit,S.cond,
                                 S.rel.crit,S_org, m_ids)
          } else {
            SS[[2]] <- "Too complex geometry"
            SS[[1]] <- NaN
          }
          if (SS[[2]] == ""){
            if (.comp(SS[[1]], best.qual)){
              .visualise_splits.good_curve(what = trace.object,
                                           level = trace.level,
                                           curve.test, SS,best.old.curve)
              best.old.curve <- curve.test
              .visualise_splits.interpol_knot(what = trace.object,
                                              level = trace.level,
                                              x = split.line.x[l+1],
                                              y = proj$x[.which_minormax(proj$y)],
                                              old.knot.y = best.y.coords[l])
              best.y.coords[l] <- test.knots[1+l]
              knot.state <- rep(TRUE, c.X.knots-2)
              knot.state[l] <- FALSE
              best.qual <- SS[[1]]
              any.qual <- c(any.qual,SS[[1]])
              #issaugom, kaip geriausia Y koordinate X koordinates knotui ta, kuri turi
              # didziausia kokybe
              best.y.knots[1+l] <- proj$x[.which_minormax(proj$y)]
            } else {
              SS[[2]] <- paste0("Poor split quality.\n","Obtained: ",
                                round(SS[[1]],2),
                                "\nRequired: ",c.sign, round(best.qual,2))
            }
          }
          if (SS[[2]] != "")
            .visualise_splits.bad_int_curve(what = trace.object,
                                            level = trace.level,
                                            curve = curve.test,
                                            x = split.line.x[l+1],
                                            y = test.knots[1+l],
                                            SS[[2]])
        }
      }
      best.y.knots[1+l] <- best.y.coords[l]
    }
    it <- it + 1
  }
  #siame etape kiekvienam X koordinates knotui turime po geriausia Y koordinate
  #sugeneruojam per siuos knotus kreive
  if (all(best.y.knots == 0 )) {
    .visualise_splits.no_best_curve(what = trace.object)
    return(list(data.frame(x = c(0,split.line.x[c.X.knots+2]),
                           y = c(0,0)),straight.qual))
  } else {
    curve.final <- stats::spline(split.line.x, best.y.knots, n=100)
    #kreive gali islisti is uz polygono (pvz. kokybes dideli iverciai buvo
    # gauti pataisius duotus knotus)
    #taigi, kreive dar karta bandoma pataisyti, jei reikia
    curve.final <- .spline_corrections(curve.final,Xup,Xdown,Yup,Ydown,best.y.knots,
                                       split.line.x,c.corr.term = c.corr.term)
    #ivertinam galutines padalinimo kreives kokybe
    environment(.curve_quality) <- environment()
    SS <- .curve_quality(curve.final, rot.poli.up, rot.poli.do, rot.dat.cords,
                         samp.dat, N.crit,
                         N.rel.crit,N.loc.crit,N.loc.rel.crit,S.cond,
                         S.rel.crit, S_org, m_ids)
    .visualise_splits.best_curve(what = trace.object,
                                 curve.final, SS)
    #grazinam padalinimo kreive, jos kokybes iverti ir visu kitu lokaliai
    # geriausiu kreiviu ivercius - SSk
    #SSk tik tam, kad pasizeti, ar tikrai grizta pati geriausia kreive
    if(length(any.qual)>0){
      if (any(.comp(any.qual,SS[[1]]))){
        warning(
          "Intermediate curves performed better than the final curve.",
          call. = FALSE
        )
      }
    }
    return(list(curve.final,SS[[1]]))
  }
}


.curvi_split<-function(knot.y.matrix,split.line.x,Xup,Xdown,Yup,Ydown,N.crit,
                       N.rel.crit,N.loc.crit,N.loc.rel.crit,S.cond,
                       S.rel.crit,S_org,c.Y.knots,c.X.knots,rot.poli.up,
                       rot.poli.do,rot.dat.cords,c.max.iter.no = c.max.iter.no,
                       c.corr.term,trace.object = trace.object,
                       trace.level = trace.level, teta = teta,
                       straight.qual,samp.dat,m_ids){
  best.y.knots <- numeric(c.X.knots)
  y.cord.quality <- rep(NA,c.Y.knots-2)
  any.qual <- numeric()
  best.y.coords <- numeric(c.X.knots-2)
  best.y.coord <- 0
  best.col.id <- 0
  old.col.id <- 0
  SS <- list(NaN,NaN)
  best.qual <- straight.qual
  best.old.curve <- data.frame(x=numeric(0),y=numeric(0))
  knot.y.matrix <- knot.y.matrix[-c(1,c.Y.knots),-c(1,c.X.knots)]
  counter <- 0
  valid.ids <- numeric()
  it <- 1
  x.seq <- (1:(c.X.knots-2))
  while(it <= c.max.iter.no) {
    if (it != 1){
      if (best.col.id == old.col.id){
        break
      } else {
        .visualise_splits.selected_knot(what = trace.object,
                                        level = trace.level,
                                        x = split.line.x[best.col.id+1],
                                        y = best.y.coord,
                                        old.knot.y = best.y.coords[best.col.id])
        best.y.coords[best.col.id] <-  best.y.coord
        best.y.knots <- c(0,best.y.coords,0)
        old.col.id <- best.col.id
        x.seq <- (1:(c.X.knots-2))[-best.col.id]
      }
    }
    for (l in x.seq) {

      for (k in 1:(c.Y.knots-2)) {

        #isbandom vis kita Y koordinate knotui su duota x koordinate
        best.y.knots[1+l] <- knot.y.matrix[k,l]
        #sugeneruojam splina per knotus
        curve <- stats::spline(split.line.x,best.y.knots,n=100)
        #iteratyviai darom, kol nebemeta klaidos:
        #patikrinam ar poligono viduj kreive, gaunam pataisos tasku koordinates
        #skeliam split.line.x ir best.y.knots i tiek daliu, kiek reikia ir
        # reikiamose vietose pridedam koordinates
        #sukuriam nauja split.line.x best.y.knots varianta
        #kartojam splina
        # if (testid == 1 & counter == 55){
        #   browser()
        #   trace.level = "all"; trace.object = "curve"}
        curve <- .spline_corrections(curve,Xup,Xdown,Yup,Ydown,best.y.knots,
                                     split.line.x,c.corr.term=c.corr.term)
        counter <- counter + 1

        if (!is.null(curve)){
          .visualise_splits.try_curve(what = trace.object,level = trace.level,
                                      counter, curve)


          #ivertinam poligono padalinimo su sugeneruota kreive kokybe
          environment(.curve_quality) <- environment()
          SS <- .curve_quality(curve=curve,rot.poli.up, rot.poli.do,
                               rot.dat.cords, samp.dat, N.crit,
                               N.rel.crit,N.loc.crit,N.loc.rel.crit,S.cond,
                               S.rel.crit,S_org,m_ids)
        } else {
          SS[[1]] <- NaN
          SS[[2]] <- "Too complex geometry"
        }



        if ( SS[[2]] != "" ){
          y.cord.quality[k] <- NaN
          message <- SS[[2]]
          .visualise_splits.bad_curve(what = trace.object,level = trace.level,
                                      curve, message)
        } else {
          #fiksuojam verte
          y.cord.quality[k] <- SS[[1]]
          any.qual <- c(any.qual,SS[[1]])
          if (.comp(SS[[1]], best.qual)){

            .visualise_splits.good_curve(what = trace.object,
                                         level = trace.level,
                                         curve, SS, best.old.curve)
            best.old.curve <- curve
            best.qual <- SS[[1]]
            best.y.coord <- knot.y.matrix[k,l]
            best.col.id <- l
            #.visualise_splits.selected_knot(what = trace.object,
            #         level = trace.level,
            #          x = split.line.x[l+1],
            #           y = knot.y.matrix[k,l],
            #            old.knot.y = best.y.coords[l])
            #best.y.coords[l] <- knot.y.matrix[k,l]
          } else {
            message <- paste0("Poor split quality.\n","Obtained: ",
                              round(SS[[1]],2),
                              "\nRequired: ",c.sign, round(best.qual,2))
            .visualise_splits.bad_curve(what = trace.object,level = trace.level,
                                        curve, message)
          }
        }
      }
      #sugeneruojam spline, kurio X asyje yra knotu Y koordinates, o Y asyje
      #padalinimo kreives kokybe
      #skirta interpoliuojant aproksimuoti tarpiniu Y koordinaciu vertes
      if (!(sum(!is.nan(y.cord.quality)) <= 3 |
            length(unique(y.cord.quality)) <= 2)){
        proj <- stats::spline(knot.y.matrix[,l], y.cord.quality, n=100)
        if (.comp(.minormax(proj$y), best.qual)){
          test.knots <- best.y.knots
          test.knots[1+l] <- proj$x[.which_minormax(proj$y)]
          curve.test <- stats::spline(split.line.x,test.knots,n=100)
          curve.test<-.spline_corrections(curve.test,Xup,Xdown,Yup,Ydown,test.knots,
                                          split.line.x,c.corr.term = c.corr.term)
          if (!is.null(curve.test)){
            .visualise_splits.try_int_curve(what = trace.object,
                                            level = trace.level,
                                            curve = curve.test,
                                            x = split.line.x[l+1],
                                            y = test.knots[1+l])
            environment(.curve_quality) <- environment()
            SS <- .curve_quality(curve.test,rot.poli.up, rot.poli.do, rot.dat.cords,
                                 samp.dat,N.crit,
                                 N.rel.crit,N.loc.crit,N.loc.rel.crit,S.cond,
                                 S.rel.crit,S_org, m_ids)
          } else {
            SS[[2]] <- "Too complex geometry"
            SS[[1]] <- NaN
          }
          if (SS[[2]] == ""){
            if (.comp(SS[[1]], best.qual)){
              .visualise_splits.good_int_curve(what = trace.object,
                                               level = trace.level,
                                               curve = curve.test,
                                               SS,
                                               best.old.curve,
                                               x = split.line.x[l+1],
                                               y = test.knots[1+l])
              best.old.curve <- curve.test
              any.qual <- c(any.qual,SS[[1]])
              #.visualise_splits.interpol_knot(what = trace.object,
              # level = trace.level,
              # x = split.line.x[l+1],
              # y = proj$x[which.max(proj$y)],
              # old.knot.y = best.y.coords[l])
              #best.y.coords[l] <- proj$x[which.max(proj$y)]
              best.y.coord <- test.knots[1+l]
              best.col.id <- l
              best.qual <- SS[[1]]
            } else {
              SS[[2]] <- paste0("Poor split quality.\n","Obtained: ",
                                round(SS[[1]],2),
                                "\nRequired: ",c.sign, round(best.qual,2))
            }
            #issaugom, kaip geriausia Y koordinate X koordinates knotui ta, kuri turi
            # didziausia kokybe
            #best.y.knots[1+l] <- proj$x[which.max(proj$y)]
          }
          if (SS[[2]] != "")
            .visualise_splits.bad_int_curve(what = trace.object,
                                            level = trace.level,
                                            curve = curve.test,
                                            x = split.line.x[l+1],
                                            y = test.knots[1+l],
                                            SS[[2]])
        } else {
          # best.y.knots[1+l] <- best.y.coords[l]
        }
      }
      best.y.knots <- c(0,best.y.coords,0)
    }

    it <- it + 1
  }
  if (it > c.max.iter.no & best.col.id != old.col.id ){
    .visualise_splits.selected_knot(what = trace.object,
                                    level = trace.level,
                                    x = split.line.x[best.col.id+1],
                                    y = best.y.coord,
                                    old.knot.y = best.y.coords[best.col.id])
    best.y.coords[best.col.id] <-  best.y.coord
    best.y.knots <- c(0,best.y.coords,0)
  }
  #siame etape kiekvienam X koordinates knotui turime po geriausia Y koordinate
  #sugeneruojam per siuos knotus kreive
  if (all(best.y.knots == 0 )) {
    .visualise_splits.no_best_curve(what = trace.object)
    return(list(data.frame(x = c(0,split.line.x[c.X.knots+2]),
                           y = c(0,0)),straight.qual))
  } else {
    curve.final <- stats::spline(split.line.x, best.y.knots, n=100)
    #kreive gali islisti is uz polygono (pvz. kokybes dideli iverciai buvo
    # gauti pataisius duotus knotus)
    #taigi, kreive dar karta bandoma pataisyti, jei reikia
    curve.final <- .spline_corrections(curve.final,Xup,Xdown,Yup,Ydown,best.y.knots,
                                       split.line.x,c.corr.term = c.corr.term)
    #ivertinam galutines padalinimo kreives kokybe
    environment(.curve_quality) <- environment()
    SS <- .curve_quality(curve.final, rot.poli.up, rot.poli.do, rot.dat.cords,
                         samp.dat, N.crit,
                         N.rel.crit,N.loc.crit,N.loc.rel.crit,S.cond,
                         S.rel.crit, S_org, m_ids)
    .visualise_splits.best_curve(what = trace.object,
                                 curve.final, SS)
    #grazinam padalinimo kreive, jos kokybes iverti ir visu kitu lokaliai
    # geriausiu kreiviu ivercius - SSk
    #SSk tik tam, kad pasizeti, ar tikrai grizta pati geriausia kreive
    if(length(any.qual)>0){
      if (any(.comp(any.qual,SS[[1]]))){
        warning(
          "Intermediate curves performed better than the final curve.",
          call. = FALSE
        )
      }
    }
    return(list(curve.final,SS[[1]]))
  }
}

#' Evaluate the quality of curve in terms of data spatial separation
#'
#' @description function forms two polygons from a provided curve (upper and lower),
#' filters data using each polygon and compares them.
#'
#' @param curve a matrix with Y coordinates in columns for each knot along the split line
#' @param rot.poli.up rotated, but not standartized, open upper polygon
#' @param rot.poli.do rotated, but not standartized, open lower polygon
#' @param rot.dat.cords rotated coordinates of data that is analyzed
#' @param samp.dat spatial subset of data
#' @param N.crit minimum number of observations required to establish
#' subdivision of a polygon.
#' @param N.rel.crit number from 0 to 0.5. The minimum allowed observation
#' proportion in polygon containing less data.
#' @param N.loc.crit Subdivision stopping criteria - number of unique locations.
#' Only meaningful when there are observations from the same location. Default
#' 0.2.
#' @param N.loc.rel.crit number from 0 to 0.5. The minimum allowed unique
#' locations proportions in polygon containing less locations.
#' @param S.cond minimum area required to establish subdivision of a plot
#' @param S.rel.crit number from 0 to 0.5. The minimum allowed area
#' proportion of a smaller polygon. Default 0.2.
#' @param S_org area of the polygon being divided.
#' @param m_ids list of data point IDs from up and bottom polygon, respectively,
#' that lie on the perimeter of polygon, but are not correctly filtered with
#' .get_ids and therefore should be added manually to the output of .get_ids.
#' @return A list of two elements: 1) rotated (not suitable for the original polygon) curve in shape of a spline that produces the best data separation; 2) quality of the division
#' @author Liudas Daumantas
#' @importFrom pracma polyarea
#' @noRd
.curve_quality<-function(curve,rot.poli.up,rot.poli.do,rot.dat.cords,
                         samp.dat,N.crit,
                         N.rel.crit,N.loc.crit,N.loc.rel.crit,S.cond,
                         S.rel.crit,S_org, m_ids){
  #sudarom poligonus padalintus kreive
  I.poli <- data.frame(x = c(rot.poli.up$x, rev(curve$x)[-1]),
                       y = c(rot.poli.up$y, rev(curve$y)[-1]))
  II.poli <- data.frame(x = c(rot.poli.do$x, rev(curve$x)[-1]),
                        y = c(rot.poli.do$y, rev(curve$y)[-1]))
  #atrenkam taskus patenkancius i skirtingas poligono dalis, padalintas kreive
  id1 <- c(.get_ids(I.poli, rot.dat.cords), m_ids[[1]])
  id2 <- c(.get_ids(II.poli,rot.dat.cords), m_ids[[2]])
  if (any(duplicated(id1)) | any(duplicated(id2))){
    warning(
      "Duplicated filtered points in curvilinear splits.",
      call. = FALSE
    )
  }
  #atliekam plotu palyginima, t.y. padalinimo gerumo ivertinima, jei kreive
  #netenkina minimaliu kriteriju - padalinimo kokybe nuline
  message <- ""
  if (N.crit > 0){
    good <- length(id1) > N.crit & length(id2) > N.crit
    if (!good){
      return(list(NA, paste0("Not enough observations.",
                             "\nObtained: ", length(id1), ' and ', length(id2),
                             "\nRequired: >", N.crit) ))
    }
  }

  if (N.rel.crit > 0){
    good <- length(id1) / nrow(rot.dat.cords) > N.rel.crit &
      length(id2) / nrow(rot.dat.cords) > N.rel.crit
    if (!good){
      return(list(NA,paste0("Too low proportion of observations.\nObtained: ",
                            round(length(id1) / nrow(rot.dat.cords), 2),
                            ' and ', round(length(id2) / nrow(rot.dat.cords),
                                           2), "\nRequired: >", N.rel.crit) ))
    }
  }

  if (N.loc.crit > 0 | N.loc.rel.crit > 0){
    n.uni.1 <- nrow(unique(rot.dat.cords[id1,]))
    n.uni.2 <- nrow(unique(rot.dat.cords[id2,]))
    if (N.loc.crit > 0){
      good <- n.uni.1 > N.loc.crit &
        n.uni.2 > N.loc.crit
      if (!good){
        return(list(NA, paste0("Not enough locations.",
                               "\nObtained: ", n.uni.1,
                               ' and ', n.uni.2,
                               "\nRequired: >", N.loc.crit) ))
      }
    }
    if (N.loc.rel.crit > 0){
      n.uni <- nrow(unique(rot.dat.cords))
      good <- n.uni.1 / n.uni > N.loc.rel.crit &
        n.uni.2 / n.uni > N.loc.rel.crit
      if (!good){
        return(list(NA, paste0("Too low proportion of locations.",
                               "\nObtained: ", round(n.uni.1 / n.uni,2),
                               ' and ', round(n.uni.2 / n.uni, 2),
                               "\nRequired: >", N.loc.rel.crit) ))
      }
    }
  }

  if (S.crit > 0 | S.rel.crit > 0){
    S1 <- abs(pracma::polyarea(I.poli$x,I.poli$y))
    S2 <- abs(pracma::polyarea(II.poli$x,II.poli$y))
    if (S.crit > 0) {
      good <- S1 > S.cond & S2 > S.cond
      if (!good){
        return(list(NA, paste0("One of the areas was too small.\n","Obtained: ",
                               round(S1,2), ' and ', round(S2,2),
                               "\nRequired: >", S.cond) ))
      }
    }
    if (S.rel.crit > 0) {
      good <- S1 / S_org  > S.rel.crit &
        S2 / S_org > S.rel.crit
      if (!good){
        message <- paste0("Too low proportion of area.\n","Obtained: ",
                          round(S1 / S_org,2), ' and ',
                          round(S2 / S_org, 2),
                          "\nRequired: >", S.rel.crit)
        return(list(NA, paste0("Too low proportion of area.\n","Obtained: ",
                               round(S1 / S_org,2), ' and ',
                               round(S2 / S_org, 2),
                               "\nRequired: >", S.rel.crit) ))
      }
    }
  }

  environment(.dif_fun) <- environment()
  SS <- .dif_fun(.slicer(samp.dat,id1),.slicer(samp.dat,id2))
  list(SS, message)
}

#' Correct the curve
#'
#' @description function corrects the curve if it is not contained within the polygon.
#' It does so by finding intervals of a curve that cross the boundaries of a polygon,
#' then it adds additional knots at the middle of these intervals inside the polygon.
#' Finally, curve is generated again using new knots, but it may still have some outlying intervals.
#' Thus, the procedure is repeated recursively until no outlying intervals are left.
#' @param curve a curve (output of spline function from stats package)
#' @param Xup X coordinates of upper part (above split line) of standartized polygon
#' @param Xdown X coordinates of lower part (below split line) of standartized polygon
#' @param Yup Y coordinates of upper part (above split line) of standartized polygon
#' @param Ydown Y coordinates of lower part (below split line) of standartized polygon
#' @param best.y.knots a vector of Y coordinates of knots of a spline that were used to produce the curve.
#' Each Y coordinate is dedicated for corresponding elements of split.line.x, that is X coordinates of spline knots.
#' @param split.line.x a vector of x coordinates for each knot along the split line
#' @param c.corr.term term that defines how much the a problematic spline
#' will be corrected (in terms of proportion of polygon width where spline
#' intersects the polygon boundary) if the spline is not contained within the
#' plot. Small values recommended (default is 0.05).
#' @param count. number of recursions.
#' @return a curve (output of a spline function from stats package)
#' @author Liudas Daumantas
#' @noRd
.spline_corrections<-function(curve,Xup,Xdown,Yup,Ydown,best.y.knots,
                             split.line.x,c.corr.term,count. = 1){

  corrected.coords<-.y_corrections(spline.x = curve$x,spline.y =curve$y ,
                                  Xup = Xup,Xdown = Xdown,Yup = Yup,
                                  Ydown = Ydown,
                                  c.corr.term = c.corr.term)
  if (length(dim(corrected.coords)) > 1){
    if (length(
      apply(corrected.coords, 1, function(xy)
        which(xy[1] == split.line.x & xy[2] == best.y.knots)))
      > 0 | count. == 25)
      return(NULL)
    count. <- count. +1
    ind<-numeric()
    k<-0
    corrected.split.line.x<-split.line.x
    corrected.best.y.knots<-best.y.knots
    for (a in seq(length(corrected.coords[,1]))){
      for (i in seq(length(split.line.x))){
        if(split.line.x[i]>=corrected.coords[a,1]){
          ind<-i-1
          corrected.split.line.x<-c(corrected.split.line.x[c(1:(ind+k))],
                                    corrected.coords[a,1],
                                    corrected.split.line.x[
                                      c((ind+k+1):
                                          length(corrected.split.line.x))
                                    ])
          corrected.best.y.knots<-c(corrected.best.y.knots[c(1:(ind+k))],
                                    corrected.coords[a,2],
                                    corrected.best.y.knots[
                                      c((ind+k+1):
                                          length(corrected.best.y.knots))
                                    ])
          k<-k+1
          break
        }
      }
    }
    curve <- tryCatch(stats::spline(corrected.split.line.x,corrected.best.y.knots,n=100),
                    warning = function(cond) NULL)
    if (is.null(curve)){
      return(curve)
    }

    best.y.knots<-corrected.best.y.knots
    split.line.x<-corrected.split.line.x
    return(.spline_corrections(curve =curve ,Xup =Xup ,Xdown =Xdown ,Yup=Yup,
                              Ydown = Ydown,best.y.knots=best.y.knots,
                              split.line.x = split.line.x,
                              c.corr.term = c.corr.term, count. = count.) )
  } else {
    return(curve)
  }
}



#' Find coordinates for new knots
#'
#' @description function checks if there are curve intervals that cross the boundary of a polygon.
#' If present, for each interval it calculates the middle X coordinate.
#' Then, Y coordinates are produced for each X coordinate in a way so that each produced Y are within the polygon.
#' The Y coordinates are calculated as follows: 1) width of the polygon is calculated
#' at X coordinate (distance in Y coord. between upper and lower boundaries of the
#' standartized polygon at X coord.); 2) the c.corr.term is then multiplied by the calculated width
#' to get the correction size in Y units; 3) correction is then added to the Y of lower polygon boundary
#' if curve crosses the boundary there, or is subtracted from the upper boundary Y if else.
#'
#' @param spline.x x coordinates of a curve
#' @param spline.y y coordinates of a curve
#' @param Xup X coordinates of upper part (above split line) of standartized polygon
#' @param Xdown X coordinates of lower part (below split line) of standartized polygon
#' @param Yup Y coordinates of upper part (above split line) of standartized polygon
#' @param Ydown Y coordinates of lower part (below split line) of standartized polygon
#' @param best.y.knots a vector of Y coordinates of knots of a spline that were used to produce the curve.
#' Each Y coordinate is dedicated for corresponding elements of split.line.x, that is X coordinates of spline knots.
#' @param c.corr.term term that defines how much the a problematic spline
#' will be corrected (in terms of proportion of polygon width where spline
#' intersects the polygon boundary) if the spline is not contained within the
#' plot. Small values recommended (default is 0.05).
#' @return If there are curve intervals that cross the boundary of a polygon,
#' then function returns a data frame of x and y coordinates that should be combined with coordinates of other knots used to produce the curve.
#' If no such intervals are present function returns "FALSE".
#' @author Liudas Daumantas
#' @noRd
.y_corrections<-function(spline.x,spline.y,Xup,Xdown,Yup,Ydown,
                        c.corr.term){
  #tikrinam, ar kreive polygone, nuimam po viena taska krastini, nes jie ant poligono ribos
  curve.in.polygon <- .point.in.polygon(
    point.x = spline.x,
    point.y = spline.y,
    pol.x = c(Xup,rev(Xdown)),
    pol.y = c(Yup,rev(Ydown))
    )[-c(1,length(spline.x))]

  #jei kreive ne polygone
  #v2 jei neatsivelgti,kuris taskas labiausiai nutoles nuo poligono
  if (any(curve.in.polygon!=1)) {
    #pasiziurim, kurie kreives taskai nepatenka i poligona
    ind<-which(curve.in.polygon!=1)
    #patikrinam, kiek yra ind verciu, jei viena, reiska, tik vienas taskas iseina is poligono, isisaugom taska
    if (length(ind)==1){
      vid.x<-spline.x[ind+1]
      vid.y<-spline.y[ind+1]
    } else { # JEI NE - gali buti kelios kreives atkarpos, nepatenkancios i poligona
      #surandam centrus intervalu, kurie nepatenka i poligona
      min.id<-ind[1]
      vid.x<-numeric()
      vid.y<-numeric()
      for (i in 1:(length(ind)-1)){
        #jei randam luzi indeksuose, reiskia priejom naujo intervalo pradzia,
        #fiksuojam buvusio intervalo viduri ir atnaujiname intervalo pradzia
        if(ind[i+1]-ind[i]>1){
          x3<-spline.x[min.id+1]+(spline.x[ind[i]+1]-spline.x[min.id+1])/2
          vid.x<-c(vid.x,x3)
          #y3 ant splino
          y3.spline<-.y.online(x=spline.x,y=spline.y,x3=x3)
          vid.y<-c(vid.y,y3.spline)
          #fiksuojam naujo intervalo pradzia
          min.id<-ind[i+1]}
      }
      # jei naujo intervalo neprieita, reiskia yra vienas intervalas - issisaugom jo centra
      if (min.id==ind[1]){
        if(ind[1]==1){
          vid.x<-spline.x[1]+(spline.x[ind[length(ind)]+1]-spline.x[1])/2
        } else {
          vid.x<-spline.x[ind[1]+1]+(spline.x[ind[length(ind)]+1]-spline.x[ind[1]+1])/2
        }
        vid.y<-.y.online(x=spline.x,y=spline.y,x3=vid.x)
      } else {# jei rasta intervalu, tai ufiksuoti ju viduriai, bet paskutinio intervalo vidurys neuzfiksuotas
        x3<-spline.x[min.id+1]+(spline.x[ind[length(ind)]+1]-spline.x[min.id+1])/2
        vid.x<-c(vid.x,x3)
        #y3 ant splino
        y3.spline<-.y.online(x=spline.x,y=spline.y,x3=x3)
        vid.y<-c(vid.y,y3.spline)
      }
    }
    #siame etape turim tasku koordinates, kuriuos reikia "nustumti" i poligona ir pakartoti splina
    #poligonas skaidytas i dvi dalis, zemiau 0 ir virs 0.
    #surandam y3 ant poligono virsaus ir apacios, x3 vietoje
    y3.up<-numeric()
    y3.down<-numeric()
    RANGES<-numeric()
    corrected.y<-numeric()
    for (i in 1:length(vid.x)){
      # are there any vertical lines in standartized polygon?
      # if so, assume that y coords of more distant (from split line)
      # vertical vertices that are the same as of vertices closer to the line.
      # crude solution...
      Yup <- .verts.y(Xup,Yup)
      Ydown <- .verts.y(Xdown,Ydown)
      y3.up<-c(y3.up,.y.online(x=Xup,y=Yup,x3=vid.x[i]))
      y3.down<-c(y3.down,.y.online(x3=vid.x[i],x=Xdown,y=Ydown))
      #surandam atstuma tarp y3 up ir down
      RANGES<-c(RANGES,y3.up[i]-y3.down[i])
      #paziurim, per kuria puse kreives atkarpa iseina uz poligono (per virsu ar ne?)
      #pataisom y3 taip, kad jis atsidurtu poligone, 5% atstumu nuo krasto poligono
      if (vid.y[i]>=0){
        corrected.y<-c(corrected.y,y3.up[i]-c.corr.term*RANGES[i])
      } else {
        corrected.y<-c(corrected.y,y3.down[i]+c.corr.term*RANGES[i])
      }
    }
    return(data.frame(vid.x,corrected.y))
  } else {
    return(FALSE)}
}

.verts.y <- function(x,y) {
  verts <- which(x[-1]-x[-length(x)] == 0)
  if (length(verts) > 0 ){
    # if so, assume that y coords of more distant (from split line)
    # vertical vertices that are the same as of vertices closer to the line.
    for (vert in verts){
      min.y.vert <- y[c(vert,vert+1)][which.min(abs(y[c(vert,vert+1)]))]
      y[c(vert,vert+1)] <- min.y.vert
    }
  }
  y
}

Try the hespdiv package in your browser

Any scripts or data that you put into this service are public.

hespdiv documentation built on May 21, 2026, 5:09 p.m.