Определите развернутые сетчатые УФ области

Предположим, у нас есть 3D-сетка с текстурными координатами для каждой вершины, поэтому, если я сделаю ее развернутой, я получу что-то вроде этого (игнорируя красный квадрат)

Сейчас я пытаюсь найти подходящий алгоритм для уникальной идентификации этих областей с использованием вершинных UV и сохранения атрибута с этим уникальным значением id. Идея состоит в том, чтобы использовать это значение в качестве индекса для таблицы цветов и получить что-то вроде этого (сделано вручную):

Я попытался перебрать каждую вершину и найти "несвязанные" треугольники, сравнивающие координаты текстуры, но порядок индексов сетки кажется не связанным с тем, как расположены UV, или я не применяю правильную формулу. У меня нет сомнений относительно того, как сохранить и передать это значение в шейдер или что-то еще, сомневаюсь, как узнать "область", к которой принадлежит вершина, или, в конечном счете, пиксель.

Благодарю.

ОБНОВЛЕНИЕ: данные, используемые для визуализации сетки, представляют собой список вершин (GL_VERTEX_BUFFER) плюс список индексов (GL_ELEMENT_ARRAY). Сетка отображается как GL_TRIANGLES, и каждая вершина является структурой, подобной этой:

struct Vertex
{
    float x, y, z;
    float nx, ny, nz;
    float tcx, tcy;
    float regionId; //the attribute I want to fill
};

struct MPUVRegionVertex
{
    float x, y;
    int faceId, regionId;
};

ОБНОВЛЕНИЕ 2: Я создал новый массив вершин MPUVRegionVertex, с элементом для каждого индекса (не для каждой уникальной вершины). После ответа @CsabaBálint я получил следующий код:

MPUVRegionVertex* uvVertexData = new MPUVRegionVertex[indexCount];

for(int ic = 0; ic < indexCount / 3; ic++)
{
    for(int vc = 0; vc < 3; vc++)
    {
        uvVertexData[3*ic+vc].x = vertexData[indexData[3*ic+vc]].tcx;
        uvVertexData[3*ic+vc].y = vertexData[indexData[3*ic+vc]].tcy;
        uvVertexData[3*ic+vc].faceId = ic;
    }
}

std::vector<std::forward_list<int> > graph(indexCount);

for(int t1=0;t1 < indexCount; ++t1)
{
    for(int t2 = t1 + 1; t2 < indexCount; ++t2)
    {
        if (uvVertexData[t1].faceId == uvVertexData[t2].faceId)
        {
            graph[t1].push_front(t2);
            graph[t2].push_front(t1);
        }
    }
}

std::forward_list<int> stack;
std::vector<int> component(indexCount);
std::set<int> notvisited;

for(int nv = 0; nv < indexCount; nv++)
{
    notvisited.insert(nv);
}


int k = 0;
while(notvisited.size() > 0)
{
    stack.push_front(*notvisited.begin());
    notvisited.erase(notvisited.begin());

    while(!stack.empty())
    {
        //SOMETHING WRONG HERE
        int temp = stack.front();
        notvisited.erase(temp);
        stack.pop_front();
        component[temp] = k;
        stack.merge(graph[temp]);
        graph[temp].clear();
    }
    k++;
}

Результатом является разное k каждые три индекса, это означает, что k++ вызывается для каждого нового треугольника, поэтому я что-то упускаю в алгоритме:S.

component[0]=0
component[1]=0
component[2]=0
component[3]=1
component[4]=1
component[5]=1
component[6]=2
component[7]=2
component[8]=2
component[9]=3
...
component[1778]=592
component[1779]=593
component[1780]=593
component[1781]=593

Некоторая информация о сетке:

Size of shape[0].indices: 1782
shape[0].positions: 1242
shape[0].texcoords: 828
shape[0].normals: 1242

ОБНОВЛЕНИЕ 3

Для получения дополнительной информации, есть только одна UV-координата для каждой вершины.

Отчисления / Правила до сих пор:

  • вершина может быть более чем в одной грани (часть более чем одного треугольника).
  • вершина будет n раз в массиве vertexToFace, один раз для каждой грани она принадлежит.
  • первая вершина в массиве vertexToFace будет произвольно иметь regionId = 0.
  • вершина принадлежит области, если она имеет те же координаты x и y или ту же грань другой вершины в этой области.

Если я правильно понял, это правильная информация для реализации нерекурсивного обхода графа. Мне нужно перебрать и сохранить как связанную, так и несвязанную вершину, все связанные вершины будут частью текущей области, все не будет проверяться снова с уже подключенными, первая итерация сохраняет первые вершины треугольника, вторая - все вершины треугольники, которые "касаются" первого треугольника, продолжаются до тех пор, пока итерация не даст новой связанной вершины (оптимизация здесь, если мы проверяем только список вершин, добавленный в последней итерации), новая добавленная вершина означает, что пришло время увеличивать regionId и начните снова с первой несвязанной вершины.

Я постараюсь реализовать поиск, следуя этому проекту.

2 ответа

Решение

Хорошо, следуя ранее упомянутым пунктам и благодаря Csaba Bálint за начальные подсказки, я нашел решение:

MPUVRegionVertex* uvVertexData = new MPUVRegionVertex[indexCount];

for(int ic = 0; ic < indexCount / 3; ic++)
{
    for(int vc = 0; vc < 3; vc++)
    {
        uvVertexData[3*ic+vc].x = vertexData[indexData[3*ic+vc]].tcx;
        uvVertexData[3*ic+vc].y = vertexData[indexData[3*ic+vc]].tcy;
        uvVertexData[3*ic+vc].faceId = ic;
    }
}

std::set<int> notAssigned;

for(int nv = 0; nv < indexCount; nv++)
{
    notAssigned.insert(nv);
}

std::forward_list<int> addedInLastIterationElements;
int currentRegion = 0;

//while there are not assigned vertex 
while (notAssigned.size() > 0)
{
    //the first not assigned element simulate that was "added" to the current region in the last check
    addedInLastIterationElements.push_front(*notAssigned.begin());

    //this first has the new current region as it's regionId
    uvVertexData[*notAssigned.begin()].regionId = currentRegion;

    //and becomes assigned
    notAssigned.erase(notAssigned.begin());

    do
    {
        //will store the elements added in the next iteration
        std::forward_list<int> newRegionElements;

        //iterate not assigned elements
        for(int currentElement : notAssigned)
        {
            //iterate elements added to the current region in the last iteration
            for(int currentRegionElement : addedInLastIterationElements)
            {
                //compare if this vertex belongs to the same region of some of the recently added
                if((uvVertexData[currentElement].x == uvVertexData[currentRegionElement].x && 
                    uvVertexData[currentElement].y == uvVertexData[currentRegionElement].y) ||
                   (uvVertexData[currentElement].faceId == uvVertexData[currentRegionElement].faceId))
                {
                    //store as an element added this iteration
                    newRegionElements.push_front(currentElement);

                    //store the current region
                    uvVertexData[currentElement].regionId = currentRegion;
                }
            }
        }

        //remove new elements from the notAssigned list
        for(int assigned : newRegionElements)
        {
            notAssigned.erase(assigned);
        }

        //replace the elements added the previous iteration for the last iteration
        addedInLastIterationElements = newRegionElements;

        //prepare for new elements
        newRegionElements.clear();
    }
    //if there is no match in the last iteration, means all remaining vertex belongs to another region
    while(!addedInLastIterationElements.empty());

    //next region
    currentRegion++;
}

Если рендеринг regionId сгенерирован с помощью таблицы цветов, я получаю желаемый результат (обратите внимание, что цвета варьируются только в 0.1 на канал, но здесь есть 10 разных цветов):

Конечно, алгоритм может быть оптимизирован, но он работает за разумное время и использует память, так что пока все в порядке.

Сейчас я пытаюсь найти подходящий алгоритм для уникальной идентификации этих областей с использованием вершинных UV и сохранения атрибута с этим уникальным значением id.

Создание графика

Сделайте идентификаторы для вершин и граней (пронумеруйте их). Но убедитесь, что одни и те же вершины получают одинаковые идентификаторы, сравните их по УФ или положению.

Создать вектор: std::vector<int> vertexToFace;

vertexToFace[i]==j означает, что i-я вершина находится на лице j.

Тогда две вершины являются соседями, если они находятся на одной грани.

Затем создайте std::vector<std::forward_list<int> > graph; Сохраните вершины как векторный индекс и добавьте соседей. (O(n^2) сложность)

Чтобы сделать это, вы должны взять i-ую вершину, и для каждого j вам нужно проверить погоду, они на одном лице. Немного оптимизированная версия:

for(int i=0; i<n; ++i) for(int j=i+1; j <n ++j)
if (vertexToFace[i] == vertexToFace[j])
{
    graph[i].push_front(j);
    graph[j].push_front(i);
}

Это O (n ^ 2), но его легко реализовать. Тяжелее, но быстрее требуется еще один вектор: std::vector<std::array<int,3>> faceToVertex;Таким образом, из i-й вершины вы можете получить доступ к ее соседям в постоянное время. В любом случае, мы построили график, в котором мы ищем связанные компоненты, что легко с поиском в глубину.

Реализация алгоритма связанных компонентов

Чтобы реализовать это, вы должны сделать еще один вектор: std::vector<bool> visited(n,false);, и еще один std::vector<int> component(n), Решение вашей проблемы будет в этом последнем.

Алгоритм прост, начать с вершины 0 и установить visited[0] = true; а также component[0]=0;, Затем для каждого невидимого соседа сделайте точно так же, как и для соседа i (некоторый элемент forward_list) if (!visited[i]) component[i] = 0;затем сделайте то же самое. Он останавливается, когда все элементы компонента становятся посещенными. Итак, вам нужно найти не посещенный элемент и повторить вышеописанное, но знать, что вы делаете компонент 1 и так далее. Пример:

int l, k=0;
while(l!=n)
{
    l=0;
    while(visited[l]) l++;
    fill_from(graph, visited, component, l, k);
    ++k;
}

Я думаю, вы поняли, так что: (псевдокод)

void fill_from(graph, visited, component, l, k)
{
    if(visited[l]) return;
    component[l] = k;
    for(auto &i : graph[l])
            fill_from(graph,visited,component,i,k);
}

Тогда мы закончили с задачей, но это еще не самое быстрое решение.

Более быстрый алгоритм

Чтобы получить еще быстрее, мы должны избавиться от рекурсии, и нам не нужен график впоследствии, используйте std::forward_list<int> для стека. Вставьте первую вершину в стек. Затем вытолкните одну вершину, установите ее компонент на k. Вставьте всех своих соседей в стек, затем удалите соседей. Другими словами, добавьте список соседей в стек (очень быстрая операция). Повторяйте до тех пор, пока стек не станет пустым.

Таким образом, мы не будем делать бесконечный цикл, потому что если мы вернемся к той же вершине, у нее не будет соседей, и мы их уже посетили. Поэтому нет необходимости в посещаемом векторе. Мы можем установить элемент вектора компонента несколько раз, но всегда на одно и то же значение, так зачем проверять его?

Однако, если у нас нет посещенного вектора, тогда будет сложнее найти другую вершину, которую мы не посетили. Хотя мы могли бы искать некоторую вершину в графе, у которой все еще есть соседи, есть лучшее решение.

Создайте std::set<int> notvisited(); Для тех точек, которые еще не посещены. Сначала он должен содержать все идентификаторы вершин, а затем каждый раз, когда мы устанавливаем идентификатор компонента, мы пытаемся удалить вершину из notvisited задавать. Мы повторяем получение вершины из набора и запускаем алгоритм fill_from(), пока набор не станет пустым, в то же время у нас будут все идентификаторы компонентов.

ОБНОВЛЕНИЕ: Использование обновленной информации о том, как хранится сетка.

Если у вас нет равных элементов в "списке вершин" (зачем вам), то индекс вершины - это ее позиция в массиве. Таким образом, идентификаторы для вершин сделаны.

Идентификаторы для треугольников или граней находятся в "списке индексов", позвольте мне назвать этот массив int listOfIndices[];для j-й грани связанные с ним вершины listOfIndices[3*j + 0], listOfIndices[3*j + 1] а также listOfIndices[3*j + 2], Чтобы сделать первый вектор, вам нужно сделать следующее:

std::vector<int> vertexToFace(num_of_verteces); //previously n
for(int j = 0; j < num_of_faces; ++j)
{
    vertexToFace[listOfIndices[3*j + 0]]=j;
    vertexToFace[listOfIndices[3*j + 1]]=j;
    vertexToFace[listOfIndices[3*j + 2]]=j;
}

(Алгоритм построения обратной зависимости); Обратите внимание, что в этом случае вам даже не нужен другой faceToVertex массив, потому что у вас уже есть (listOfIndices), вы просто должны индексировать его по-разному (делить на 3 каждый раз).

Другие вопросы по тегам