ccls/src/position.cc
2017-11-18 09:27:37 -08:00

169 lines
3.7 KiB
C++

#include "position.h"
namespace {
// Skips until the character immediately following |skip_after|.
const char* SkipAfter(const char* input, char skip_after) {
while (*input && *input != skip_after)
++input;
++input;
return input;
}
} // namespace
Position::Position() {}
Position::Position(int16_t line, int16_t column) : line(line), column(column) {}
Position::Position(const char* encoded) {
assert(encoded);
line = (int16_t)atoi(encoded);
encoded = SkipAfter(encoded, ':');
assert(encoded);
column = (int16_t)atoi(encoded);
}
std::string Position::ToString() {
// Output looks like this:
//
// 1:2
//
// 1 => line
// 2 => column
std::string result;
result += std::to_string(line);
result += ':';
result += std::to_string(column);
return result;
}
std::string Position::ToPrettyString(const std::string& filename) {
// Output looks like this:
//
// 1:2:3
//
// 1 => filename
// 2 => line
// 3 => column
std::string result;
result += filename;
result += ':';
result += std::to_string(line);
result += ':';
result += std::to_string(column);
return result;
}
bool Position::operator==(const Position& that) const {
return line == that.line && column == that.column;
}
bool Position::operator!=(const Position& that) const {
return !(*this == that);
}
bool Position::operator<(const Position& that) const {
if (line < that.line)
return true;
return line == that.line && column < that.column;
}
Range::Range() {}
Range::Range(Position position) : Range(position, position) {}
Range::Range(Position start, Position end) : start(start), end(end) {}
Range::Range(const char* encoded) {
end = start;
start.line = (int16_t)atoi(encoded);
encoded = SkipAfter(encoded, ':');
assert(encoded);
start.column = (int16_t)atoi(encoded);
encoded = SkipAfter(encoded, '-');
assert(encoded);
end.line = (int16_t)atoi(encoded);
encoded = SkipAfter(encoded, ':');
assert(encoded);
end.column = (int16_t)atoi(encoded);
}
bool Range::Contains(int line, int column) const {
if (line == start.line && line == end.line)
return column >= start.column && column < end.column;
if (line == start.line)
return column >= start.column;
if (line == end.line)
return column < end.column;
if (line > start.line && line < end.line)
return true;
return false;
}
std::string Range::ToString() {
// Output looks like this:
//
// *1:2-3:4
//
// * => if present, range is interesting
// 1 => start line
// 2 => start column
// 3 => end line
// 4 => end column
std::string output;
output += std::to_string(start.line);
output += ':';
output += std::to_string(start.column);
output += '-';
output += std::to_string(end.line);
output += ':';
output += std::to_string(end.column);
return output;
}
bool Range::operator==(const Range& that) const {
return start == that.start && end == that.end;
}
bool Range::operator!=(const Range& that) const {
return !(*this == that);
}
bool Range::operator<(const Range& that) const {
if (start < that.start)
return true;
return start == that.start && end < that.end;
}
// Position
void Reflect(Reader& visitor, Position& value) {
if (!visitor.IsString())
value = Position();
else
value = Position(visitor.GetString());
}
void Reflect(Writer& visitor, Position& value) {
std::string output = value.ToString();
visitor.String(output.c_str(), (rapidjson::SizeType)output.size());
}
// Range
void Reflect(Reader& visitor, Range& value) {
if (!visitor.IsString())
value = Range();
else
value = Range(visitor.GetString());
}
void Reflect(Writer& visitor, Range& value) {
std::string output = value.ToString();
visitor.String(output.c_str(), (rapidjson::SizeType)output.size());
}