package syntax

import (
	
	
	
	
	
	
)

// CharSet combines start-end rune ranges and unicode categories representing a set of characters
type CharSet struct {
	ranges     []singleRange
	categories []category
	sub        *CharSet //optional subtractor
	negate     bool
	anything   bool
}

type category struct {
	negate bool
	cat    string
}

type singleRange struct {
	first rune
	last  rune
}

const (
	spaceCategoryText = " "
	wordCategoryText  = "W"
)

var (
	ecmaSpace = []rune{0x0009, 0x000e, 0x0020, 0x0021, 0x00a0, 0x00a1, 0x1680, 0x1681, 0x2000, 0x200b, 0x2028, 0x202a, 0x202f, 0x2030, 0x205f, 0x2060, 0x3000, 0x3001, 0xfeff, 0xff00}
	ecmaWord  = []rune{0x0030, 0x003a, 0x0041, 0x005b, 0x005f, 0x0060, 0x0061, 0x007b}
	ecmaDigit = []rune{0x0030, 0x003a}

	re2Space = []rune{0x0009, 0x000b, 0x000c, 0x000e, 0x0020, 0x0021}
)

var (
	AnyClass          = getCharSetFromOldString([]rune{0}, false)
	ECMAAnyClass      = getCharSetFromOldString([]rune{0, 0x000a, 0x000b, 0x000d, 0x000e}, false)
	NoneClass         = getCharSetFromOldString(nil, false)
	ECMAWordClass     = getCharSetFromOldString(ecmaWord, false)
	NotECMAWordClass  = getCharSetFromOldString(ecmaWord, true)
	ECMASpaceClass    = getCharSetFromOldString(ecmaSpace, false)
	NotECMASpaceClass = getCharSetFromOldString(ecmaSpace, true)
	ECMADigitClass    = getCharSetFromOldString(ecmaDigit, false)
	NotECMADigitClass = getCharSetFromOldString(ecmaDigit, true)

	WordClass     = getCharSetFromCategoryString(false, false, wordCategoryText)
	NotWordClass  = getCharSetFromCategoryString(true, false, wordCategoryText)
	SpaceClass    = getCharSetFromCategoryString(false, false, spaceCategoryText)
	NotSpaceClass = getCharSetFromCategoryString(true, false, spaceCategoryText)
	DigitClass    = getCharSetFromCategoryString(false, false, "Nd")
	NotDigitClass = getCharSetFromCategoryString(false, true, "Nd")

	RE2SpaceClass    = getCharSetFromOldString(re2Space, false)
	NotRE2SpaceClass = getCharSetFromOldString(re2Space, true)
)

var unicodeCategories = func() map[string]*unicode.RangeTable {
	 := make(map[string]*unicode.RangeTable)
	for ,  := range unicode.Scripts {
		[] = 
	}
	for ,  := range unicode.Categories {
		[] = 
	}
	for ,  := range unicode.Properties {
		[] = 
	}
	return 
}()

func getCharSetFromCategoryString( bool,  bool,  ...string) func() *CharSet {
	if  &&  {
		panic("BUG!  You should only negate the set OR the category in a constant setup, but not both")
	}

	 := CharSet{negate: }

	.categories = make([]category, len())
	for ,  := range  {
		.categories[] = category{cat: , negate: }
	}
	return func() *CharSet {
		//make a copy each time
		 := 
		//return that address
		return &
	}
}

func getCharSetFromOldString( []rune,  bool) func() *CharSet {
	 := CharSet{}
	if len() > 0 {
		 := false
		 := len()
		if  {
			if [0] == 0 {
				 = [1:]
			} else {
				++
				 = true
			}
		}

		if %2 == 0 {
			.ranges = make([]singleRange, /2)
		} else {
			.ranges = make([]singleRange, /2+1)
		}

		 := true
		if  {
			.ranges[0] = singleRange{first: 0}
			 = false
		}

		 := 0
		for ,  := range  {
			if  {
				// lower bound in a new range
				.ranges[] = singleRange{first: }
				 = false
			} else {
				.ranges[].last =  - 1
				++
				 = true
			}
		}
		if ! {
			.ranges[].last = utf8.MaxRune
		}
	}

	return func() *CharSet {
		 := 
		return &
	}
}

// Copy makes a deep copy to prevent accidental mutation of a set
func ( CharSet) () CharSet {
	 := CharSet{
		anything: .anything,
		negate:   .negate,
	}

	.ranges = append(.ranges, .ranges...)
	.categories = append(.categories, .categories...)

	if .sub != nil {
		 := .sub.()
		.sub = &
	}

	return 
}

// gets a human-readable description for a set string
func ( CharSet) () string {
	 := &bytes.Buffer{}
	.WriteRune('[')

	if .IsNegated() {
		.WriteRune('^')
	}

	for ,  := range .ranges {

		.WriteString(CharDescription(.first))
		if .first != .last {
			if .last-.first != 1 {
				//groups that are 1 char apart skip the dash
				.WriteRune('-')
			}
			.WriteString(CharDescription(.last))
		}
	}

	for ,  := range .categories {
		.WriteString(.String())
	}

	if .sub != nil {
		.WriteRune('-')
		.WriteString(.sub.())
	}

	.WriteRune(']')

	return .String()
}

// mapHashFill converts a charset into a buffer for use in maps
func ( CharSet) ( *bytes.Buffer) {
	if .negate {
		.WriteByte(0)
	} else {
		.WriteByte(1)
	}

	binary.Write(, binary.LittleEndian, len(.ranges))
	binary.Write(, binary.LittleEndian, len(.categories))
	for ,  := range .ranges {
		.WriteRune(.first)
		.WriteRune(.last)
	}
	for ,  := range .categories {
		.WriteString(.cat)
		if .negate {
			.WriteByte(1)
		} else {
			.WriteByte(0)
		}
	}

	if .sub != nil {
		.sub.()
	}
}

// CharIn returns true if the rune is in our character set (either ranges or categories).
// It handles negations and subtracted sub-charsets.
func ( CharSet) ( rune) bool {
	 := false
	// in s && !s.subtracted

	//check ranges
	for ,  := range .ranges {
		if  < .first {
			continue
		}
		if  <= .last {
			 = true
			break
		}
	}

	//check categories if we haven't already found a range
	if ! && len(.categories) > 0 {
		for ,  := range .categories {
			// special categories...then unicode
			if .cat == spaceCategoryText {
				if unicode.IsSpace() {
					// we found a space so we're done
					// negate means this is a "bad" thing
					 = !.negate
					break
				} else if .negate {
					 = true
					break
				}
			} else if .cat == wordCategoryText {
				if IsWordChar() {
					 = !.negate
					break
				} else if .negate {
					 = true
					break
				}
			} else if unicode.Is(unicodeCategories[.cat], ) {
				// if we're in this unicode category then we're done
				// if negate=true on this category then we "failed" our test
				// otherwise we're good that we found it
				 = !.negate
				break
			} else if .negate {
				 = true
				break
			}
		}
	}

	// negate the whole char set
	if .negate {
		 = !
	}

	// get subtracted recurse
	if  && .sub != nil {
		 = !.sub.()
	}

	//log.Printf("Char '%v' in %v == %v", string(ch), c.String(), val)
	return 
}

func ( category) () string {
	switch .cat {
	case spaceCategoryText:
		if .negate {
			return "\\S"
		}
		return "\\s"
	case wordCategoryText:
		if .negate {
			return "\\W"
		}
		return "\\w"
	}
	if ,  := unicodeCategories[.cat];  {

		if .negate {
			return "\\P{" + .cat + "}"
		}
		return "\\p{" + .cat + "}"
	}
	return "Unknown category: " + .cat
}

// CharDescription Produces a human-readable description for a single character.
func ( rune) string {
	/*if ch == '\\' {
		return "\\\\"
	}

	if ch > ' ' && ch <= '~' {
		return string(ch)
	} else if ch == '\n' {
		return "\\n"
	} else if ch == ' ' {
		return "\\ "
	}*/

	 := &bytes.Buffer{}
	escape(, , false) //fmt.Sprintf("%U", ch)
	return .String()
}

// According to UTS#18 Unicode Regular Expressions (http://www.unicode.org/reports/tr18/)
// RL 1.4 Simple Word Boundaries  The class of <word_character> includes all Alphabetic
// values from the Unicode character database, from UnicodeData.txt [UData], plus the U+200C
// ZERO WIDTH NON-JOINER and U+200D ZERO WIDTH JOINER.
func ( rune) bool {
	//"L", "Mn", "Nd", "Pc"
	return unicode.In(,
		unicode.Categories["L"], unicode.Categories["Mn"],
		unicode.Categories["Nd"], unicode.Categories["Pc"]) ||  == '\u200D' ||  == '\u200C'
	//return 'A' <= r && r <= 'Z' || 'a' <= r && r <= 'z' || '0' <= r && r <= '9' || r == '_'
}

func ( rune) bool {
	return unicode.In(,
		unicode.Categories["L"], unicode.Categories["Mn"],
		unicode.Categories["Nd"], unicode.Categories["Pc"])

	//return 'A' <= r && r <= 'Z' || 'a' <= r && r <= 'z' || '0' <= r && r <= '9' || r == '_'
}

// SingletonChar will return the char from the first range without validation.
// It assumes you have checked for IsSingleton or IsSingletonInverse and will panic given bad input
func ( CharSet) () rune {
	return .ranges[0].first
}

func ( CharSet) () bool {
	return !.negate && //negated is multiple chars
		len(.categories) == 0 && len(.ranges) == 1 && // multiple ranges and unicode classes represent multiple chars
		.sub == nil && // subtraction means we've got multiple chars
		.ranges[0].first == .ranges[0].last // first and last equal means we're just 1 char
}

func ( CharSet) () bool {
	return .negate && //same as above, but requires negated
		len(.categories) == 0 && len(.ranges) == 1 && // multiple ranges and unicode classes represent multiple chars
		.sub == nil && // subtraction means we've got multiple chars
		.ranges[0].first == .ranges[0].last // first and last equal means we're just 1 char
}

func ( CharSet) () bool {
	return !.IsNegated() && !.HasSubtraction()
}

func ( CharSet) () bool {
	return .negate
}

func ( CharSet) () bool {
	return .sub != nil
}

func ( CharSet) () bool {
	return len(.ranges) == 0 && len(.categories) == 0 && .sub == nil
}

func ( *CharSet) (,  bool,  string) {
	if  {
		if  {
			.addRanges(NotECMADigitClass().ranges)
		} else {
			.addRanges(ECMADigitClass().ranges)
		}
	} else {
		.addCategories(category{cat: "Nd", negate: })
	}
}

func ( *CharSet) ( rune) {
	.addRange(, )
}

func ( *CharSet) (, ,  bool) {
	if  {
		if  {
			.addRanges(NotECMASpaceClass().ranges)
		} else {
			.addRanges(ECMASpaceClass().ranges)
		}
	} else if  {
		if  {
			.addRanges(NotRE2SpaceClass().ranges)
		} else {
			.addRanges(RE2SpaceClass().ranges)
		}
	} else {
		.addCategories(category{cat: spaceCategoryText, negate: })
	}
}

func ( *CharSet) (,  bool) {
	if  {
		if  {
			.addRanges(NotECMAWordClass().ranges)
		} else {
			.addRanges(ECMAWordClass().ranges)
		}
	} else {
		.addCategories(category{cat: wordCategoryText, negate: })
	}
}

// Add set ranges and categories into ours -- no deduping or anything
func ( *CharSet) ( CharSet) {
	if .anything {
		return
	}
	if .anything {
		.makeAnything()
		return
	}
	// just append here to prevent double-canon
	.ranges = append(.ranges, .ranges...)
	.addCategories(.categories...)
	.canonicalize()
}

func ( *CharSet) () {
	.anything = true
	.categories = []category{}
	.ranges = AnyClass().ranges
}

func ( *CharSet) ( ...category) {
	// don't add dupes and remove positive+negative
	if .anything {
		// if we've had a previous positive+negative group then
		// just return, we're as broad as we can get
		return
	}

	for ,  := range  {
		 := false
		for ,  := range .categories {
			if .cat == .cat {
				if .negate != .negate {
					// oposite negations...this mean we just
					// take us as anything and move on
					.makeAnything()
					return
				}
				 = true
				break
			}
		}

		if ! {
			.categories = append(.categories, )
		}
	}
}

// Merges new ranges to our own
func ( *CharSet) ( []singleRange) {
	if .anything {
		return
	}
	.ranges = append(.ranges, ...)
	.canonicalize()
}

// Merges everything but the new ranges into our own
func ( *CharSet) ( []singleRange) {
	if .anything {
		return
	}

	var  rune

	// convert incoming ranges into opposites, assume they are in order
	for ,  := range  {
		if  < .first {
			.ranges = append(.ranges, singleRange{, .first - 1})
		}
		 = .last + 1
	}

	if  < utf8.MaxRune {
		.ranges = append(.ranges, singleRange{, utf8.MaxRune})
	}

	.canonicalize()
}

func isValidUnicodeCat( string) bool {
	,  := unicodeCategories[]
	return 
}

func ( *CharSet) ( string, ,  bool,  string) {
	if !isValidUnicodeCat() {
		// unknown unicode category, script, or property "blah"
		panic(fmt.Errorf("Unknown unicode category, script, or property '%v'", ))

	}

	if  && ( == "Ll" ||  == "Lu" ||  == "Lt") {
		// when RegexOptions.IgnoreCase is specified then {Ll} {Lu} and {Lt} cases should all match
		.addCategories(
			category{cat: "Ll", negate: },
			category{cat: "Lu", negate: },
			category{cat: "Lt", negate: })
	}
	.addCategories(category{cat: , negate: })
}

func ( *CharSet) ( *CharSet) {
	.sub = 
}

func ( *CharSet) (,  rune) {
	.ranges = append(.ranges, singleRange{first: , last: })
	.canonicalize()
}

func ( *CharSet) ( string,  bool) bool {
	var  []singleRange

	switch  {
	case "alnum":
		 = []singleRange{singleRange{'0', '9'}, singleRange{'A', 'Z'}, singleRange{'a', 'z'}}
	case "alpha":
		 = []singleRange{singleRange{'A', 'Z'}, singleRange{'a', 'z'}}
	case "ascii":
		 = []singleRange{singleRange{0, 0x7f}}
	case "blank":
		 = []singleRange{singleRange{'\t', '\t'}, singleRange{' ', ' '}}
	case "cntrl":
		 = []singleRange{singleRange{0, 0x1f}, singleRange{0x7f, 0x7f}}
	case "digit":
		.addDigit(false, , "")
	case "graph":
		 = []singleRange{singleRange{'!', '~'}}
	case "lower":
		 = []singleRange{singleRange{'a', 'z'}}
	case "print":
		 = []singleRange{singleRange{' ', '~'}}
	case "punct": //[!-/:-@[-`{-~]
		 = []singleRange{singleRange{'!', '/'}, singleRange{':', '@'}, singleRange{'[', '`'}, singleRange{'{', '~'}}
	case "space":
		.addSpace(true, false, )
	case "upper":
		 = []singleRange{singleRange{'A', 'Z'}}
	case "word":
		.addWord(true, )
	case "xdigit":
		 = []singleRange{singleRange{'0', '9'}, singleRange{'A', 'F'}, singleRange{'a', 'f'}}
	default:
		return false
	}

	if len() > 0 {
		if  {
			.addNegativeRanges()
		} else {
			.addRanges()
		}
	}

	return true
}

type singleRangeSorter []singleRange

func ( singleRangeSorter) () int           { return len() }
func ( singleRangeSorter) (,  int) bool { return [].first < [].first }
func ( singleRangeSorter) (,  int)      { [], [] = [], [] }

// Logic to reduce a character class to a unique, sorted form.
func ( *CharSet) () {
	var ,  int
	var  rune

	//
	// Find and eliminate overlapping or abutting ranges
	//

	if len(.ranges) > 1 {
		sort.Sort(singleRangeSorter(.ranges))

		 := false

		for ,  = 1, 0; ; ++ {
			for  = .ranges[].last; ; ++ {
				if  == len(.ranges) ||  == utf8.MaxRune {
					 = true
					break
				}

				 := .ranges[]
				if .first > +1 {
					break
				}

				if  < .last {
					 = .last
				}
			}

			.ranges[] = singleRange{first: .ranges[].first, last: }

			++

			if  {
				break
			}

			if  <  {
				.ranges[] = .ranges[]
			}
		}

		.ranges = append(.ranges[:], .ranges[len(.ranges):]...)
	}
}

// Adds to the class any lowercase versions of characters already
// in the class. Used for case-insensitivity.
func ( *CharSet) () {
	if .anything {
		return
	}
	 := []singleRange{}
	for  := 0;  < len(.ranges); ++ {
		 := .ranges[]
		if .first == .last {
			 := unicode.ToLower(.first)
			.ranges[] = singleRange{first: , last: }
		} else {
			 = append(, )
		}
	}

	for ,  := range  {
		.addLowercaseRange(.first, .last)
	}
	.canonicalize()
}

/**************************************************************************
    Let U be the set of Unicode character values and let L be the lowercase
    function, mapping from U to U. To perform case insensitive matching of
    character sets, we need to be able to map an interval I in U, say

        I = [chMin, chMax] = { ch : chMin <= ch <= chMax }

    to a set A such that A contains L(I) and A is contained in the union of
    I and L(I).

    The table below partitions U into intervals on which L is non-decreasing.
    Thus, for any interval J = [a, b] contained in one of these intervals,
    L(J) is contained in [L(a), L(b)].

    It is also true that for any such J, [L(a), L(b)] is contained in the
    union of J and L(J). This does not follow from L being non-decreasing on
    these intervals. It follows from the nature of the L on each interval.
    On each interval, L has one of the following forms:

        (1) L(ch) = constant            (LowercaseSet)
        (2) L(ch) = ch + offset         (LowercaseAdd)
        (3) L(ch) = ch | 1              (LowercaseBor)
        (4) L(ch) = ch + (ch & 1)       (LowercaseBad)

    It is easy to verify that for any of these forms [L(a), L(b)] is
    contained in the union of [a, b] and L([a, b]).
***************************************************************************/

const (
	LowercaseSet = 0 // Set to arg.
	LowercaseAdd = 1 // Add arg.
	LowercaseBor = 2 // Bitwise or with 1.
	LowercaseBad = 3 // Bitwise and with 1 and add original.
)

type lcMap struct {
	chMin, chMax rune
	op, data     int32
}

var lcTable = []lcMap{
	lcMap{'\u0041', '\u005A', LowercaseAdd, 32},
	lcMap{'\u00C0', '\u00DE', LowercaseAdd, 32},
	lcMap{'\u0100', '\u012E', LowercaseBor, 0},
	lcMap{'\u0130', '\u0130', LowercaseSet, 0x0069},
	lcMap{'\u0132', '\u0136', LowercaseBor, 0},
	lcMap{'\u0139', '\u0147', LowercaseBad, 0},
	lcMap{'\u014A', '\u0176', LowercaseBor, 0},
	lcMap{'\u0178', '\u0178', LowercaseSet, 0x00FF},
	lcMap{'\u0179', '\u017D', LowercaseBad, 0},
	lcMap{'\u0181', '\u0181', LowercaseSet, 0x0253},
	lcMap{'\u0182', '\u0184', LowercaseBor, 0},
	lcMap{'\u0186', '\u0186', LowercaseSet, 0x0254},
	lcMap{'\u0187', '\u0187', LowercaseSet, 0x0188},
	lcMap{'\u0189', '\u018A', LowercaseAdd, 205},
	lcMap{'\u018B', '\u018B', LowercaseSet, 0x018C},
	lcMap{'\u018E', '\u018E', LowercaseSet, 0x01DD},
	lcMap{'\u018F', '\u018F', LowercaseSet, 0x0259},
	lcMap{'\u0190', '\u0190', LowercaseSet, 0x025B},
	lcMap{'\u0191', '\u0191', LowercaseSet, 0x0192},
	lcMap{'\u0193', '\u0193', LowercaseSet, 0x0260},
	lcMap{'\u0194', '\u0194', LowercaseSet, 0x0263},
	lcMap{'\u0196', '\u0196', LowercaseSet, 0x0269},
	lcMap{'\u0197', '\u0197', LowercaseSet, 0x0268},
	lcMap{'\u0198', '\u0198', LowercaseSet, 0x0199},
	lcMap{'\u019C', '\u019C', LowercaseSet, 0x026F},
	lcMap{'\u019D', '\u019D', LowercaseSet, 0x0272},
	lcMap{'\u019F', '\u019F', LowercaseSet, 0x0275},
	lcMap{'\u01A0', '\u01A4', LowercaseBor, 0},
	lcMap{'\u01A7', '\u01A7', LowercaseSet, 0x01A8},
	lcMap{'\u01A9', '\u01A9', LowercaseSet, 0x0283},
	lcMap{'\u01AC', '\u01AC', LowercaseSet, 0x01AD},
	lcMap{'\u01AE', '\u01AE', LowercaseSet, 0x0288},
	lcMap{'\u01AF', '\u01AF', LowercaseSet, 0x01B0},
	lcMap{'\u01B1', '\u01B2', LowercaseAdd, 217},
	lcMap{'\u01B3', '\u01B5', LowercaseBad, 0},
	lcMap{'\u01B7', '\u01B7', LowercaseSet, 0x0292},
	lcMap{'\u01B8', '\u01B8', LowercaseSet, 0x01B9},
	lcMap{'\u01BC', '\u01BC', LowercaseSet, 0x01BD},
	lcMap{'\u01C4', '\u01C5', LowercaseSet, 0x01C6},
	lcMap{'\u01C7', '\u01C8', LowercaseSet, 0x01C9},
	lcMap{'\u01CA', '\u01CB', LowercaseSet, 0x01CC},
	lcMap{'\u01CD', '\u01DB', LowercaseBad, 0},
	lcMap{'\u01DE', '\u01EE', LowercaseBor, 0},
	lcMap{'\u01F1', '\u01F2', LowercaseSet, 0x01F3},
	lcMap{'\u01F4', '\u01F4', LowercaseSet, 0x01F5},
	lcMap{'\u01FA', '\u0216', LowercaseBor, 0},
	lcMap{'\u0386', '\u0386', LowercaseSet, 0x03AC},
	lcMap{'\u0388', '\u038A', LowercaseAdd, 37},
	lcMap{'\u038C', '\u038C', LowercaseSet, 0x03CC},
	lcMap{'\u038E', '\u038F', LowercaseAdd, 63},
	lcMap{'\u0391', '\u03AB', LowercaseAdd, 32},
	lcMap{'\u03E2', '\u03EE', LowercaseBor, 0},
	lcMap{'\u0401', '\u040F', LowercaseAdd, 80},
	lcMap{'\u0410', '\u042F', LowercaseAdd, 32},
	lcMap{'\u0460', '\u0480', LowercaseBor, 0},
	lcMap{'\u0490', '\u04BE', LowercaseBor, 0},
	lcMap{'\u04C1', '\u04C3', LowercaseBad, 0},
	lcMap{'\u04C7', '\u04C7', LowercaseSet, 0x04C8},
	lcMap{'\u04CB', '\u04CB', LowercaseSet, 0x04CC},
	lcMap{'\u04D0', '\u04EA', LowercaseBor, 0},
	lcMap{'\u04EE', '\u04F4', LowercaseBor, 0},
	lcMap{'\u04F8', '\u04F8', LowercaseSet, 0x04F9},
	lcMap{'\u0531', '\u0556', LowercaseAdd, 48},
	lcMap{'\u10A0', '\u10C5', LowercaseAdd, 48},
	lcMap{'\u1E00', '\u1EF8', LowercaseBor, 0},
	lcMap{'\u1F08', '\u1F0F', LowercaseAdd, -8},
	lcMap{'\u1F18', '\u1F1F', LowercaseAdd, -8},
	lcMap{'\u1F28', '\u1F2F', LowercaseAdd, -8},
	lcMap{'\u1F38', '\u1F3F', LowercaseAdd, -8},
	lcMap{'\u1F48', '\u1F4D', LowercaseAdd, -8},
	lcMap{'\u1F59', '\u1F59', LowercaseSet, 0x1F51},
	lcMap{'\u1F5B', '\u1F5B', LowercaseSet, 0x1F53},
	lcMap{'\u1F5D', '\u1F5D', LowercaseSet, 0x1F55},
	lcMap{'\u1F5F', '\u1F5F', LowercaseSet, 0x1F57},
	lcMap{'\u1F68', '\u1F6F', LowercaseAdd, -8},
	lcMap{'\u1F88', '\u1F8F', LowercaseAdd, -8},
	lcMap{'\u1F98', '\u1F9F', LowercaseAdd, -8},
	lcMap{'\u1FA8', '\u1FAF', LowercaseAdd, -8},
	lcMap{'\u1FB8', '\u1FB9', LowercaseAdd, -8},
	lcMap{'\u1FBA', '\u1FBB', LowercaseAdd, -74},
	lcMap{'\u1FBC', '\u1FBC', LowercaseSet, 0x1FB3},
	lcMap{'\u1FC8', '\u1FCB', LowercaseAdd, -86},
	lcMap{'\u1FCC', '\u1FCC', LowercaseSet, 0x1FC3},
	lcMap{'\u1FD8', '\u1FD9', LowercaseAdd, -8},
	lcMap{'\u1FDA', '\u1FDB', LowercaseAdd, -100},
	lcMap{'\u1FE8', '\u1FE9', LowercaseAdd, -8},
	lcMap{'\u1FEA', '\u1FEB', LowercaseAdd, -112},
	lcMap{'\u1FEC', '\u1FEC', LowercaseSet, 0x1FE5},
	lcMap{'\u1FF8', '\u1FF9', LowercaseAdd, -128},
	lcMap{'\u1FFA', '\u1FFB', LowercaseAdd, -126},
	lcMap{'\u1FFC', '\u1FFC', LowercaseSet, 0x1FF3},
	lcMap{'\u2160', '\u216F', LowercaseAdd, 16},
	lcMap{'\u24B6', '\u24D0', LowercaseAdd, 26},
	lcMap{'\uFF21', '\uFF3A', LowercaseAdd, 32},
}

func ( *CharSet) (,  rune) {
	var , ,  int
	var ,  rune
	var  lcMap

	for ,  = 0, len(lcTable);  < ; {
		 = ( + ) / 2
		if lcTable[].chMax <  {
			 =  + 1
		} else {
			 = 
		}
	}

	for ;  < len(lcTable); ++ {
		 = lcTable[]
		if .chMin >  {
			return
		}
		 = .chMin
		if  <  {
			 = 
		}

		 = .chMax
		if  >  {
			 = 
		}

		switch .op {
		case LowercaseSet:
			 = rune(.data)
			 = rune(.data)
			break
		case LowercaseAdd:
			 += .data
			 += .data
			break
		case LowercaseBor:
			 |= 1
			 |= 1
			break
		case LowercaseBad:
			 += ( & 1)
			 += ( & 1)
			break
		}

		if  <  ||  >  {
			.addRange(, )
		}
	}
}