R/sfc_sequence.R

Defines functions c.sfc_sequence length.sfc_sequence `[<-.sfc_sequence` `[.sfc_sequence` next_rotation sfc_unit sfc_seed sfc_sequence

Documented in c.sfc_sequence length.sfc_sequence sfc_seed sfc_sequence sfc_unit

#' Constructor of the sfc_sequence class
#' 
#' @param seq A sequence of base patterns. The value can be a vector of letters or a single string.
#' @param rot The corresponding rotations of base patterns. If it has length one and the sequence contains R/L/I (right/left/forward),
#'          `rot` controls the rotation of the first base pattern and the rotations for remaining base patterns in the sequence are automatically
#'          calculated.
#' @param universe The universe of base patterns. A vector of letters.
#' 
#' @details
#' This funtion is very low-level. Normally, users do not need to directly use this constructor.
#' 
#' @export
#' @return An `sfc_sequence` object.
#' @examples
#' sfc_sequence("ABCD", rot = c(0, 90, 180, 270), universe = c("A", "B", "C", "D"))
sfc_sequence = function(seq, rot = 0L, universe = NULL) {

	if(is.factor(seq)) {
		universe = levels(seq)
	} else {
		if(length(seq) == 1) {
			seq = strsplit(seq, "")[[1]]
		}
		if(is.null(universe)) {
			seq = factor(seq)
			universe = levels(seq)
		} else {
			seq = factor(seq, levels = universe)
		}
	}

	n = length(seq)

	rot = as.integer(rot)
	if(length(rot) == 1) {
		rot2 = integer(n)
		rot2[1] = rot[1]
		if(n > 1) {
			for(i in 2:n) {
				rot2[i] = next_rotation(seq[i-1], rot2[i-1])
			}
		}
		rot = rot2
	}

	diff = setdiff(levels(seq), universe)
	if(length(diff)) {
		stop_wrap("all letters should be included in universe.")
	}
	
	p = new("sfc_sequence")
	p@seq = seq
	p@rot = rot %% 360L
	p@universe = universe


	p
}

#' @rdname sfc_sequence
#' @details
#' `sfc_seed` class is the same as the `sfc_sequence` class. It is used specifically as the "seed sequence"
#' when generating the curves. 
#' @export
sfc_seed = function(seq, rot = 0L, universe = NULL) {
	p = sfc_sequence(seq, rot, universe)
	p2 = new("sfc_seed")
	p2@seq = p@seq
	p2@rot = p@rot
	p2@universe = p@universe
	p2
}



#' @rdname sfc_sequence
#' @details
#' `sfc_unit` class also inherits the `sfc_sequence` class but has one additionally slot: `corner`. 
#' It is used specifically when defining the expansion rules. 
#' @export
sfc_unit = function(seq, rot = 0L, universe = NULL) {
	p = sfc_sequence(seq, rot, universe)

	p2 = new("sfc_unit")
	p2@seq = p@seq
	p2@rot = p@rot
	p2@universe = p@universe

	p2
}



# rotation of the direction of approaching
next_rotation = function(letter, rot) {
	if(letter == "R") {
		return(rot - 90L)
	} else if(letter == "L") {
		return(rot + 90L)
	} else if(letter == "U") {
		return(rot + 180L)
	} else {
		return(rot)
	}
}


# subset allows an incomplete curve

#' Utility functions
#' @rdname utility
#' @param x An `sfc_sequence` object.
#' @param i Numeric index.
#' @param value An `sfc_sequence` object.
#' @param ... A list of `sfc_sequence` objects or other arguments.
#' 
#' @export
#' @examples
#' p = sfc_sequence("ABCDEFGH")
#' p
#' p[1:4]
#' p[1:4] = p[8:5]
#' p
#' length(p)
#' c(p, p)
`[.sfc_sequence` = function(x, i) {
	x2 = new("sfc_sequence")
	x2@seq = x@seq[i]
	x2@rot = x@rot[i]
	x2@universe = x@universe
	x2
}


#' @rdname utility
#' @export
`[<-.sfc_sequence` = function(x, i, value) {
	if(sfc_is_compatible(x, value, strict = FALSE)) {
		x@seq[i] = value@seq
		x@rot[i] = value@rot
		x
	} else {
		stop_wrap("`x` and `value` should be compatible.")
	}
}

#' @rdname utility
#' @return 
#' `length.sfc_sequence()` returns an integer scalar.
#' @export
length.sfc_sequence = function(x) {
	length(x@seq)
}

#' @rdname utility
#' @details
#' For efficiency, `c.sfc_sequence()` does not check whether the input `sfc_sequence` objects are compatible.
#' @return
#' `c.sfc_sequence()` returns an `sfc_sequence` object.
#' @export
c.sfc_sequence = function(...) {
	lt = list(...)
	seq = do.call("c", lapply(lt, function(x) x@seq))
	rot = do.call("c", lapply(lt, function(x) x@rot))
	
	p = new("sfc_sequence")
	p@seq = seq
	p@rot = rot
	p@universe = lt[[1]]@universe

	p
}


#' @param object The corresponding object.
#' @rdname show
#' @export
#' @return No value is returned.
setMethod("show",
	signature = "sfc_sequence",
	definition = function(object) {

	n = length(object)
	nr = ceiling(n/8)

	seq = object@seq
	rot = object@rot

	rot_str = function(x) {
		# ifelse(x == 0, "  0", ifelse(x == 90, " 90", x))
		x
	}

	cat("A sequence of ", n, " base pattern", ifelse(n == 1, ".", "s."), "\n", sep = "")
	if(nr > 4) {
		other_lines = paste0(".... other ", nr-4, " line", ifelse(nr-4 > 1, "s", ""), " ....")

		df = matrix("", nrow = 4, ncol = 2)
		for(i in 1:2) {
			ind = seq( (i-1)*8+1, i*8 )
			df[i, 1] = paste(seq[ ind[1:4] ], "(", rot_str(rot[ ind[1:4] ]), ")", sep = "", collapse = "")
			df[i, 2] = paste(seq[ ind[5:8] ], "(", rot_str(rot[ ind[5:8] ]), ")", sep = "", collapse = "")
		}

		ind = seq( (nr-2)*8+1, (nr-1)*8 )
		df[3, 1] = paste(seq[ ind[1:4] ], "(", rot_str(rot[ ind[1:4] ]), ")", sep = "", collapse = "")
		df[3, 2] = paste(seq[ ind[5:8] ], "(", rot_str(rot[ ind[5:8] ]), ")", sep = "", collapse = "")

		ind = seq( (nr-1)*8+1, min((nr)*8, n) )
		if(length(ind) <= 4) {
			df[4, 1] = paste(seq[ ind ], "(", rot_str(rot[ ind ]), ")", sep = "", collapse = "")
			df[4, 2] = ""
		} else if(length(ind) < 8) {
			df[4, 1] = paste(seq[ ind[1:4] ], "(", rot_str(rot[ ind[1:4] ]), ")", sep = "", collapse = "")
			df[4, 2] = paste(seq[ ind[5:length(ind)] ], "(", rot_str(rot[ ind[5:length(ind)] ]), ")", sep = "", collapse = "")
		} else {
			df[4, 1] = paste(seq[ ind[1:4] ], "(", rot_str(rot[ ind[1:4] ]), ")", sep = "", collapse = "")
			df[4, 2] = paste(seq[ ind[5:8] ], "(", rot_str(rot[ ind[5:8] ]), ")", sep = "", collapse = "")
		}

		max_nchar1 = max(nchar(df[, 1]))
		max_nchar2 = max(nchar(df[, 2]))

		for(i in 1:2) {
			cat("  ")
			cat(df[i, 1]); cat(strrep(" ", max_nchar1 - nchar(df[i, 1])))
			cat("  ")
			cat(df[i, 2]); cat("\n")
		}
		cat("  ", other_lines, "\n", sep = "")
		for(i in 3:4) {
			cat("  ")
			cat(df[i, 1]); cat(strrep(" ", max_nchar1 - nchar(df[i, 1])))
			cat("  ")
			cat(df[i, 2]); cat("\n")
		}

	} else {
		df = matrix("", nrow = nr, ncol = 2)
		for(i in 1:nr) {
			ind = seq( (i-1)*8+1, min(i*8, n) )

			if(length(ind) <= 4) {
				df[i, 1] = paste(seq[ ind ], "(", rot_str(rot[ ind ]), ")", sep = "", collapse = "")
				df[i, 2] = ""
			} else if(length(ind) < 8) {
				df[i, 1] = paste(seq[ ind[1:4] ], "(", rot_str(rot[ ind[1:4] ]), ")", sep = "", collapse = "")
				df[i, 2] = paste(seq[ ind[5:length(ind)] ], "(", rot_str(rot[ ind[5:length(ind)] ]), ")", sep = "", collapse = "")
			} else {
				df[i, 1] = paste(seq[ ind[1:4] ], "(", rot_str(rot[ ind[1:4] ]), ")", sep = "", collapse = "")
				df[i, 2] = paste(seq[ ind[5:8] ], "(", rot_str(rot[ ind[5:8] ]), ")", sep = "", collapse = "")
			}
		}

		max_nchar1 = max(nchar(df[, 1]))
		max_nchar2 = max(nchar(df[, 2]))

		for(i in 1:nr) {
			cat("  ")
			cat(df[i, 1]); cat(strrep(" ", max_nchar1 - nchar(df[i, 1])))
			cat("  ")
			cat(df[i, 2]); cat("\n")
		}
	}
})

#' @rdname sfc_universe
#' @export
setMethod("sfc_universe",
	signature = "sfc_sequence",
	definition = function(p) {
	p@universe
})

#' Whether two sfc_sequence objects are compatible
#' @aliases sfc_is_compatible
#' @rdname sfc_is_compatible
#' @param p1 An `sfc_sequence` object.
#' @param p2 An `sfc_sequence` object.
#' @param strict `TRUE` or `FALSE`, see **Details**.
#' 
#' @details
#' The function compares whether the two universe base pattern sets are identical.
#' If `strict` is `TRUE`, the order of the two universe sets should also be the same.
#' If `strict` is `FALSE`, the universe set of `p2` can be a subset of the universe set of `p1`.
#' @export
#' @return A logical scalar.
#' @examples
#' p1 = sfc_2x2("I")
#' p2 = sfc_2x2("R")
#' sfc_is_compatible(p1, p2)
#' 
#' p1 = sfc_2x2("I")
#' p2 = sfc_sequence("R")
#' sfc_is_compatible(p1, p2)
#' sfc_is_compatible(p1, p2, strict = FALSE)
#' 
#' p1 = sfc_sequence("ABC")
#' p2 = sfc_sequence("DEF")
#' sfc_is_compatible(p1, p2)
setMethod("sfc_is_compatible",
	signature = c("sfc_sequence", "sfc_sequence"),
	definition = function(p1, p2, strict = TRUE) {
	if(strict) {
		identical(sfc_universe(p1), sfc_universe(p2))
	} else {
		length(setdiff(sfc_universe(p2), sfc_universe(p1))) == 0
	}
})

#' @rdname sfc_is_compatible
#' @export
setMethod("sfc_is_compatible",
	signature = c("sfc_sequence", "sfc_rules"),
	definition = function(p1, p2) {
	identical(sfc_universe(p1), sfc_universe(p2))
})

#' @rdname sfc_is_compatible
#' @export
setMethod("sfc_is_compatible",
	signature = c("sfc_rules", "sfc_sequence"),
	definition = function(p1, p2) {
	identical(sfc_universe(p1), sfc_universe(p2))
})

Try the sfcurve package in your browser

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

sfcurve documentation built on Sept. 14, 2024, 1:07 a.m.