修改泳道接口,添加多个操作函数
This commit is contained in:
parent
5f06ddc447
commit
a51a1284f2
@ -16,47 +16,53 @@ public:
|
||||
~PBLane();
|
||||
|
||||
private:
|
||||
//泳道过滤处理函数
|
||||
//根据输入所有类似泳道的中心X轴坐标(groupsX),以及泳道的最小最大距离(minDifference/maxDifference),过滤掉不合规的泳道
|
||||
std::vector<int> processArray(std::vector<int>& groupsX, int minDifference, int maxDifference);
|
||||
|
||||
//宽度/*流*/计算函数,用来计算检测泳道宽度流
|
||||
vector<int> get_bar_num(vector<unsigned char> buf);
|
||||
|
||||
//泳道检测补全函数
|
||||
//根据泳道位置(processX)和泳道间距离(meanW),从图像(src)中检测临近类泳道位置是否符合泳道特性(泳道内两条像素的宽度流应类似且存在条带)
|
||||
std::vector<int> checkArray(std::vector<int>& processX, Mat src, int start_y, int end_y, int meanW);
|
||||
|
||||
//条带信息计算函数,输入8bit数组(rbuf)以及过滤范围(range)(此范围内仅保留一个分子)
|
||||
vector<std::array<int, 3>> get_top_point(vector<unsigned char> rbuf, int range);
|
||||
|
||||
//单个泳道中条带信息计算(条带位置等)
|
||||
BandInfo get_protein_lane_data(Mat src, Rect lane);
|
||||
static bool myCompare(const int& a, const int& b);
|
||||
static bool myCompare2(const std::array<int, 4>& a, const std::array<int, 4>& b);
|
||||
|
||||
|
||||
public:
|
||||
//函数:获得图像中泳道位置
|
||||
//src:输入图像,需是8bit单通道图像
|
||||
//返回值:泳道矩形位置信息
|
||||
std::vector<cv::Rect> getProteinRect(Mat src);
|
||||
|
||||
////函数:获得图像中泳道对应的条带信息
|
||||
////src:输入图像,需是16bit单通道图像
|
||||
////lanes:泳道矩形位置信息
|
||||
////返回值:对应条带位置信息
|
||||
// std::vector<cv::Rect> getProteinRect(Mat src);
|
||||
//获得蛋白泳道位置,src是灰度图像CV_8UC1,*ProteinRect_width 是蛋白泳道宽度(keep_width = 1时,输入固定宽度;keep_width = 0时,输出计算得到宽度),
|
||||
//ProteinRect_height_ratio高度比例,高度占图像高度百分之几,一般输入90
|
||||
//返回所有蛋白泳道矩形
|
||||
std::vector<cv::Rect> getProteinRect(Mat src,int* ProteinRect_width,bool keep_width,int ProteinRect_height_ratio);
|
||||
//添加泳道,proteinRect是当前泳道矩形,x是新添加泳道的中心x坐标,一般就是鼠标点击图形的x坐标,
|
||||
//src是输入图形(CV_16UC1),unadjustbands是未对齐条带信息,需要同步修改
|
||||
void addProteinRect(std::vector<cv::Rect>& proteinRect,int x,Mat src,std::vector<BandInfo>& unadjustbands);
|
||||
//删除泳道,proteinRect是当前泳道矩形,idx是删除泳道的下标序号,unadjustbands是未对齐条带信息,需要同步修改
|
||||
void deleteProteinRect(std::vector<cv::Rect>& proteinRect,int idx,std::vector<BandInfo>& unadjustbands);
|
||||
//计算单个泳道条带信息,src是输入图形(CV_16UC1),lane是泳道对应矩形,
|
||||
//返回BandInfo条带结果
|
||||
BandInfo get_protein_lane_data(Mat src,Rect lane);
|
||||
//获得所有泳道未对齐的条带结果,src是输入图形(CV_16UC1),lane是泳道对应矩形
|
||||
//返回所有泳道未对齐的条带结果
|
||||
std::vector<BandInfo> getProteinBands(Mat src,std::vector<cv::Rect> lanes);
|
||||
|
||||
////函数:条带分子匹配排序函数
|
||||
////bands:所有条带位置信息
|
||||
////range:Y轴差在range范围内的表示和泳道1匹配
|
||||
////返回值:结果修改在bands中
|
||||
void adjustBands(std::vector<BandInfo>& bands, int range);
|
||||
|
||||
|
||||
////函数:计算所有条带的分子信息
|
||||
////lanes:输入泳道信息
|
||||
////bands:输入条带信息,对应输出的分子信息在这个结构体中
|
||||
//修改泳道宽度和高度和已经计算出的未对齐条带信息,src是输入图形(CV_16UC1),proteinRect是当前泳道矩形,
|
||||
//ProteinRect_width是新的蛋白泳道宽度,ProteinRect_height_ratio是新的高度比例,高度占图像高度百分之几,一般输入90
|
||||
//unadjustbands是未对齐的所有泳道条带信息
|
||||
//输入的条带必须是未对齐的,调整后数据需要重新对齐和计算分子量(调用adjustBands和molecularWeightResult)
|
||||
void modifyProteinRectAndBands(Mat src,std::vector<cv::Rect>& proteinRect,int ProteinRect_width,int ProteinRect_height_ratio,std::vector<BandInfo>& unadjustbands);
|
||||
//根据鼠标点击图形的坐标,输出当前位置的泳道下标和条带下标
|
||||
//lanes是泳道信息,bands是对应条带信息(不关心对齐或者未对齐),x、y是坐标,lanesIndex是传回鼠标点击位置的泳道下标,bandsIndex是鼠标点击位置的条带下标
|
||||
void getLaneBandsIndex(std::vector<cv::Rect> lanes,std::vector<BandInfo> bands,int x,int y,int* lanesIndex,int* bandsIndex);
|
||||
//根据鼠标点击图形的坐标添加条带,lanes是泳道信息,lanesIndex是鼠标点击位置是第几条泳道,unadjustbands是未对齐的所有泳道条带信息,原始鼠标点击位置的y坐标
|
||||
//输入的条带必须是未对齐的,调整后数据需要重新对齐和计算分子量(调用adjustBands和molecularWeightResult)
|
||||
void addProteinBand(std::vector<cv::Rect> lanes,int lanesIndex,std::vector<BandInfo>& unadjustbands,int y);
|
||||
//根据鼠标点击图形的坐标删除条带,lanesIndex是鼠标点击位置是第几条泳道,unadjustbands是未对齐的所有泳道条带信息,bandsIndex是鼠标点击位置是第几个条带
|
||||
//输入的条带必须是未对齐的,调整后数据需要重新对齐和计算分子量(调用adjustBands和molecularWeightResult)
|
||||
void deleteProteinBand(int lanesIndex,std::vector<BandInfo>& unadjustbands,int bandsIndex);
|
||||
//对齐所有泳道的条带结果,unadjustbands是所有泳道的条带结果,range是条带中心在多少像素范围内属于同一类型条带,一般写10
|
||||
//返回对齐后的所有泳道的条带结果(未计算分子量)
|
||||
std::vector<BandInfo> adjustBands(std::vector<BandInfo> unadjustbands, int range);
|
||||
//计算得到分子量结果
|
||||
void molecularWeightResult(std::vector<cv::Rect> lanes,std::vector<BandInfo>& bands);
|
||||
|
||||
};
|
@ -259,7 +259,7 @@ std::vector<int> PBLane::checkArray(std::vector<int>& processX, Mat src, int sta
|
||||
return result;
|
||||
}
|
||||
|
||||
vector<std::array<int, 3>> PBLane::get_top_point(vector<unsigned char> rbuf, int range)
|
||||
vector<std::array<int, 3>> PBLane::get_top_point(vector<unsigned short> rbuf, int range)
|
||||
{
|
||||
vector<std::array<int, 2>> point;
|
||||
vector<char> diff(rbuf.size());
|
||||
@ -279,7 +279,7 @@ vector<std::array<int, 3>> PBLane::get_top_point(vector<unsigned char> rbuf, int
|
||||
}
|
||||
for (int i = rbuf.size() - 1; i >= 0; i--)
|
||||
{
|
||||
if (diff[i] == 0 && i == rbuf.size() - 1)
|
||||
if (i == rbuf.size() - 1 && diff[i] == 0)
|
||||
{
|
||||
diff[i] = 1;
|
||||
}
|
||||
@ -437,37 +437,12 @@ vector<std::array<int, 3>> PBLane::get_top_point(vector<unsigned char> rbuf, int
|
||||
return topPoint;
|
||||
}
|
||||
|
||||
BandInfo PBLane::get_protein_lane_data(Mat src, Rect lane)
|
||||
{
|
||||
BandInfo band;
|
||||
int sum = 0;
|
||||
vector<unsigned char> cdata;
|
||||
for (int i = lane.y; i < lane.y + lane.height; i++)
|
||||
{
|
||||
sum = 0;
|
||||
for (int j = lane.x; j < lane.x + lane.width; j++)
|
||||
{
|
||||
sum += src.at<unsigned short>(i, j);
|
||||
}
|
||||
sum /= lane.width;
|
||||
band.land_data.push_back((unsigned short)sum);
|
||||
band.ydata.push_back((float)((65535 - sum) / 255.0));
|
||||
cdata.push_back((unsigned char)((65535 - sum) >> 8));
|
||||
}
|
||||
for (int i = 0; i < lane.height; i++)
|
||||
{
|
||||
band.xdata.push_back((float)i / lane.height);
|
||||
}
|
||||
band.band_point = get_top_point(cdata, 8);
|
||||
return band;
|
||||
}
|
||||
|
||||
|
||||
PBLane::PBLane() {}
|
||||
|
||||
PBLane::~PBLane() {}
|
||||
|
||||
std::vector<cv::Rect> PBLane::getProteinRect(Mat src)
|
||||
std::vector<cv::Rect> PBLane::getProteinRect(Mat src,int* ProteinRect_width,bool keep_width,int ProteinRect_height_ratio)
|
||||
{
|
||||
cv::Mat edges;
|
||||
Canny(src, edges, 50, 150);
|
||||
@ -506,15 +481,16 @@ std::vector<cv::Rect> PBLane::getProteinRect(Mat src)
|
||||
int meanW = 0;
|
||||
int meanY = 0;
|
||||
int maxH = 0;
|
||||
std::sort(allW.begin(), allW.end(), myCompare);
|
||||
std::sort(allY.begin(), allY.end(), myCompare);
|
||||
std::sort(allX.begin(), allX.end(), myCompare);
|
||||
std::sort(allW.begin(), allW.end(), [](const int& a, const int& b) {
|
||||
return a < b;
|
||||
});
|
||||
std::sort(allY.begin(), allY.end(), [](const int& a, const int& b) {
|
||||
return a < b;
|
||||
});
|
||||
std::sort(allX.begin(), allX.end(), [](const int& a, const int& b) {
|
||||
return a < b;
|
||||
});
|
||||
int cnt = 0;
|
||||
if (boundingRects.size() == 0)
|
||||
{
|
||||
std::vector<cv::Rect> proteinRect1(boundingRects.size());
|
||||
return proteinRect1;
|
||||
}
|
||||
for(int i = boundingRects.size()/3;i <= 2*boundingRects.size()/3;i++)
|
||||
{
|
||||
cnt++;
|
||||
@ -546,15 +522,77 @@ std::vector<cv::Rect> PBLane::getProteinRect(Mat src)
|
||||
std::vector<cv::Rect> proteinRect(proteinX.size());
|
||||
for(int i = 0;i<proteinX.size();i++)
|
||||
{
|
||||
if(keep_width == 1){
|
||||
meanW = *ProteinRect_width;
|
||||
}
|
||||
else{
|
||||
*ProteinRect_width = meanW;
|
||||
}
|
||||
int y_start = src.rows * (100 - ProteinRect_height_ratio) / 200;
|
||||
proteinRect[i].x = proteinX[i] - meanW/2;
|
||||
|
||||
proteinRect[i].y = 15;//meanY - maxH/2;
|
||||
proteinRect[i].y = y_start;
|
||||
proteinRect[i].width = meanW;
|
||||
proteinRect[i].height = src.rows - 30;
|
||||
proteinRect[i].height = src.rows - y_start * 2;
|
||||
}
|
||||
return proteinRect;
|
||||
}
|
||||
void PBLane::addProteinRect(std::vector<cv::Rect>& proteinRect,int x,Mat src,std::vector<BandInfo>& unadjustbands)
|
||||
{
|
||||
Rect new_proteinRect;
|
||||
new_proteinRect.x = x - proteinRect[0].width/2;
|
||||
new_proteinRect.y = proteinRect[0].y;
|
||||
new_proteinRect.width = proteinRect[0].width;
|
||||
new_proteinRect.height = proteinRect[0].height;
|
||||
BandInfo band = get_protein_lane_data(src,new_proteinRect);
|
||||
|
||||
for (auto it = proteinRect.begin(); it != proteinRect.end(); ++it) {
|
||||
if (it->x > new_proteinRect.x) {
|
||||
size_t index = std::distance(proteinRect.begin(), it);
|
||||
proteinRect.insert(it, new_proteinRect);
|
||||
unadjustbands.insert(unadjustbands.begin() + index, band);
|
||||
return;
|
||||
}
|
||||
}
|
||||
proteinRect.push_back(new_proteinRect);
|
||||
unadjustbands.push_back(band);
|
||||
return;
|
||||
}
|
||||
|
||||
void PBLane::deleteProteinRect(std::vector<cv::Rect>& proteinRect,int idx,std::vector<BandInfo>& unadjustbands)
|
||||
{
|
||||
if (idx < proteinRect.size()) {
|
||||
proteinRect.erase(proteinRect.begin() + idx);
|
||||
unadjustbands.erase(unadjustbands.begin() + idx);
|
||||
} else {
|
||||
std::cout << "idx out of range!" << std::endl;
|
||||
}
|
||||
return;
|
||||
}
|
||||
|
||||
BandInfo PBLane::get_protein_lane_data(Mat src, Rect lane)
|
||||
{
|
||||
BandInfo band;
|
||||
int sum = 0;
|
||||
vector<unsigned short> cdata;
|
||||
for (int i = lane.y; i < lane.y + lane.height; i++)
|
||||
{
|
||||
sum = 0;
|
||||
for (int j = lane.x; j < lane.x + lane.width; j++)
|
||||
{
|
||||
sum += src.at<unsigned short>(i, j);
|
||||
}
|
||||
sum /= lane.width;
|
||||
band.land_data.push_back((unsigned short)sum);
|
||||
band.ydata.push_back((float)((65535 - sum) / 255.0));
|
||||
cdata.push_back(65535 - sum);
|
||||
}
|
||||
for (int i = 0; i < lane.height; i++)
|
||||
{
|
||||
band.xdata.push_back((float)i / lane.height);
|
||||
}
|
||||
band.band_point = get_top_point(cdata, 8);
|
||||
return band;
|
||||
}
|
||||
|
||||
std::vector<BandInfo> PBLane::getProteinBands(Mat src, std::vector<cv::Rect> lanes)
|
||||
{
|
||||
@ -580,11 +618,198 @@ std::vector<BandInfo> PBLane::getProteinBands(Mat src, std::vector<cv::Rect> lan
|
||||
return bands;
|
||||
}
|
||||
|
||||
|
||||
void PBLane::adjustBands(std::vector<BandInfo>& bands, int range)
|
||||
void PBLane::modifyProteinRectAndBands(Mat src,std::vector<cv::Rect>& proteinRect,int ProteinRect_width,int ProteinRect_height_ratio,std::vector<BandInfo>& unadjustbands)
|
||||
{
|
||||
// 用于存储所有的值及其对应的列索引
|
||||
int y_start = src.rows * (100 - ProteinRect_height_ratio) / 200;
|
||||
int det_y = proteinRect[0].y - y_start;
|
||||
for(int i = 0;i<proteinRect.size();i++)
|
||||
{
|
||||
proteinRect[i].x = proteinRect[i].x - (ProteinRect_width - proteinRect[i].width)/2;
|
||||
proteinRect[i].y = y_start;
|
||||
proteinRect[i].width = ProteinRect_width;
|
||||
proteinRect[i].height = src.rows - y_start * 2;
|
||||
}
|
||||
int sum = 0;
|
||||
for(int n = 0; n < proteinRect.size(); n++)
|
||||
{
|
||||
unadjustbands[n].land_data.clear();
|
||||
unadjustbands[n].ydata.clear();
|
||||
unadjustbands[n].xdata.clear();
|
||||
for(int i = proteinRect[n].y; i < proteinRect[n].y + proteinRect[n].height; i++)
|
||||
{
|
||||
sum = 0;
|
||||
for(int j = proteinRect[n].x; j < proteinRect[n].x + proteinRect[n].width; j++)
|
||||
{
|
||||
sum+=src.at<unsigned short>(i,j);
|
||||
}
|
||||
sum/=proteinRect[n].width;
|
||||
unadjustbands[n].land_data.push_back((unsigned short)sum);
|
||||
unadjustbands[n].ydata.push_back((float)((65535 - sum)/255.0));
|
||||
}
|
||||
for(int i = 0; i < proteinRect[n].height; i++)
|
||||
{
|
||||
unadjustbands[n].xdata.push_back((float)i/proteinRect[n].height);
|
||||
}
|
||||
|
||||
for(int i = 0; i < unadjustbands[n].band_point.size(); i++)
|
||||
{
|
||||
//fix left and right
|
||||
unadjustbands[n].band_point[i][1] += det_y;
|
||||
unadjustbands[n].band_point[i][2] += det_y;
|
||||
//fix top
|
||||
int y1 = unadjustbands[n].band_point[i][1];
|
||||
int y2 = unadjustbands[n].band_point[i][2];
|
||||
float temp = unadjustbands[n].ydata[y1];
|
||||
for(int j = y1 + 1; j < y2; j++)
|
||||
{
|
||||
if(unadjustbands[n].ydata[j] > temp){
|
||||
temp = unadjustbands[n].ydata[j];
|
||||
unadjustbands[n].band_point[i][0] = j;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
void PBLane::getLaneBandsIndex(std::vector<cv::Rect> lanes,std::vector<BandInfo> bands,int x,int y,int* lanesIndex,int* bandsIndex)
|
||||
{
|
||||
*lanesIndex = -1;
|
||||
*bandsIndex = -1;
|
||||
int ystart = lanes[0].y;
|
||||
int yend = lanes[0].y + lanes[0].height - 1;
|
||||
if(y > yend || y < ystart)
|
||||
{
|
||||
return ;
|
||||
}
|
||||
for (int i = 0; i < lanes.size(); i++) {
|
||||
int xstart = lanes[i].x;
|
||||
int xend = lanes[i].x + lanes[i].width - 1;
|
||||
if(x <= xend && x >= xstart){
|
||||
*lanesIndex = i;
|
||||
break;
|
||||
}
|
||||
}
|
||||
for (int j = 0; j < bands[*lanesIndex].band_point.size(); j++) {
|
||||
int y1 = bands[*lanesIndex].band_point[j][1] + ystart;
|
||||
int y2 = bands[*lanesIndex].band_point[j][2] + ystart;
|
||||
if(y <= y2 && y >= y1){
|
||||
*bandsIndex = j;
|
||||
break;
|
||||
}
|
||||
}
|
||||
return;
|
||||
}
|
||||
|
||||
void PBLane::addProteinBand(std::vector<cv::Rect> lanes,int lanesIndex,std::vector<BandInfo>& unadjustbands,int y)
|
||||
{
|
||||
int ystart = lanes[lanesIndex].y;
|
||||
int yend = lanes[lanesIndex].y + lanes[lanesIndex].height - 1;
|
||||
int y1 = 0;
|
||||
int y2 = lanes[lanesIndex].height;
|
||||
if(y < ystart || y > yend)
|
||||
{
|
||||
return ;
|
||||
}
|
||||
int rect_y = y - lanes[lanesIndex].y;
|
||||
for(int i = 0; i < unadjustbands[lanesIndex].band_point.size(); i++)
|
||||
{
|
||||
y2 = unadjustbands[lanesIndex].band_point[i][1];
|
||||
if(rect_y >= y1 && rect_y < y2)
|
||||
{
|
||||
break;
|
||||
}
|
||||
else
|
||||
{
|
||||
y1 = unadjustbands[lanesIndex].band_point[i][2];
|
||||
}
|
||||
|
||||
}
|
||||
if(y1 > y2)
|
||||
{
|
||||
y2 = lanes[lanesIndex].height;
|
||||
}
|
||||
|
||||
//fine top
|
||||
int range = 8;
|
||||
int top_start = rect_y - range;
|
||||
int top_end = rect_y + range;
|
||||
if(top_start < y1){
|
||||
top_start = y1;
|
||||
}
|
||||
if(top_end > y2){
|
||||
top_end = y2;
|
||||
}
|
||||
std::array<int, 3> new_band_point;
|
||||
new_band_point[0] = top_start;
|
||||
float temp = unadjustbands[lanesIndex].ydata[top_start];
|
||||
float mean = unadjustbands[lanesIndex].ydata[top_start];
|
||||
for(int i = top_start + 1; i < top_end; i++)
|
||||
{
|
||||
mean += unadjustbands[lanesIndex].ydata[i];
|
||||
if(unadjustbands[lanesIndex].ydata[i] > temp){
|
||||
temp = unadjustbands[lanesIndex].ydata[i];
|
||||
new_band_point[0] = i;
|
||||
}
|
||||
}
|
||||
mean /= (top_end - top_start);
|
||||
|
||||
//fine left
|
||||
temp = unadjustbands[lanesIndex].ydata[new_band_point[0]];
|
||||
new_band_point[1] = y1;
|
||||
for(int i = new_band_point[0] - 1; i > y1; i--)
|
||||
{
|
||||
if(unadjustbands[lanesIndex].ydata[i] > temp)
|
||||
{
|
||||
if(temp < mean)
|
||||
{
|
||||
new_band_point[1] = i+1;
|
||||
break;
|
||||
}
|
||||
}
|
||||
temp = unadjustbands[lanesIndex].ydata[i];
|
||||
}
|
||||
//fine right
|
||||
temp = unadjustbands[lanesIndex].ydata[new_band_point[0]];
|
||||
new_band_point[2] = y2;
|
||||
for(int i = new_band_point[0] + 1; i < y2; i++)
|
||||
{
|
||||
if(unadjustbands[lanesIndex].ydata[i] > temp)
|
||||
{
|
||||
if(temp < mean)
|
||||
{
|
||||
new_band_point[2] = i+1;
|
||||
break;
|
||||
}
|
||||
}
|
||||
temp = unadjustbands[lanesIndex].ydata[i];
|
||||
}
|
||||
|
||||
for (auto it = unadjustbands[lanesIndex].band_point.begin(); it != unadjustbands[lanesIndex].band_point.end(); ++it) {
|
||||
if ((*it)[0] > new_band_point[0]) {
|
||||
unadjustbands[lanesIndex].band_point.insert(it, new_band_point);
|
||||
return;
|
||||
}
|
||||
}
|
||||
unadjustbands[lanesIndex].band_point.push_back(new_band_point);
|
||||
return ;
|
||||
}
|
||||
|
||||
void PBLane::deleteProteinBand(int lanesIndex,std::vector<BandInfo>& unadjustbands,int bandsIndex)
|
||||
{
|
||||
if (bandsIndex < unadjustbands[lanesIndex].band_point.size()) {
|
||||
unadjustbands[lanesIndex].band_point.erase(unadjustbands[lanesIndex].band_point.begin() + bandsIndex);
|
||||
} else {
|
||||
std::cout << "bandsIndex out of range!" << std::endl;
|
||||
}
|
||||
return;
|
||||
}
|
||||
|
||||
|
||||
std::vector<BandInfo> PBLane::adjustBands(std::vector<BandInfo> unadjustbands, int range)
|
||||
{
|
||||
// 用于存储所有的值及其对应的列索引
|
||||
std::map<int, int> columnMap;
|
||||
std::vector<BandInfo> bands = unadjustbands;
|
||||
|
||||
std::vector<std::array<int, 4>> vec;
|
||||
for (int i = 0; i < bands.size(); i++)
|
||||
|
Loading…
Reference in New Issue
Block a user