Spaces:
Runtime error
Runtime error
| const cv::Size MaskedImage::kDownsampleKernelSize = cv::Size(6, 6); | |
| const int MaskedImage::kDownsampleKernel[6] = {1, 5, 10, 10, 5, 1}; | |
| bool MaskedImage::contains_mask(int y, int x, int patch_size) const { | |
| auto mask_size = size(); | |
| for (int dy = -patch_size; dy <= patch_size; ++dy) { | |
| for (int dx = -patch_size; dx <= patch_size; ++dx) { | |
| int yy = y + dy, xx = x + dx; | |
| if (yy >= 0 && yy < mask_size.height && xx >= 0 && xx < mask_size.width) { | |
| if (is_masked(yy, xx) && !is_globally_masked(yy, xx)) return true; | |
| } | |
| } | |
| } | |
| return false; | |
| } | |
| MaskedImage MaskedImage::downsample() const { | |
| const auto &kernel_size = MaskedImage::kDownsampleKernelSize; | |
| const auto &kernel = MaskedImage::kDownsampleKernel; | |
| const auto size = this->size(); | |
| const auto new_size = cv::Size(size.width / 2, size.height / 2); | |
| auto ret = MaskedImage(new_size.width, new_size.height); | |
| if (!m_global_mask.empty()) ret.init_global_mask_mat(); | |
| for (int y = 0; y < size.height - 1; y += 2) { | |
| for (int x = 0; x < size.width - 1; x += 2) { | |
| int r = 0, g = 0, b = 0, ksum = 0; | |
| bool is_gmasked = true; | |
| for (int dy = -kernel_size.height / 2 + 1; dy <= kernel_size.height / 2; ++dy) { | |
| for (int dx = -kernel_size.width / 2 + 1; dx <= kernel_size.width / 2; ++dx) { | |
| int yy = y + dy, xx = x + dx; | |
| if (yy >= 0 && yy < size.height && xx >= 0 && xx < size.width) { | |
| if (!is_globally_masked(yy, xx)) { | |
| is_gmasked = false; | |
| } | |
| if (!is_masked(yy, xx)) { | |
| auto source_ptr = get_image(yy, xx); | |
| int k = kernel[kernel_size.height / 2 - 1 + dy] * kernel[kernel_size.width / 2 - 1 + dx]; | |
| r += source_ptr[0] * k, g += source_ptr[1] * k, b += source_ptr[2] * k; | |
| ksum += k; | |
| } | |
| } | |
| } | |
| } | |
| if (ksum > 0) r /= ksum, g /= ksum, b /= ksum; | |
| if (!m_global_mask.empty()) { | |
| ret.set_global_mask(y / 2, x / 2, is_gmasked); | |
| } | |
| if (ksum > 0) { | |
| auto target_ptr = ret.get_mutable_image(y / 2, x / 2); | |
| target_ptr[0] = r, target_ptr[1] = g, target_ptr[2] = b; | |
| ret.set_mask(y / 2, x / 2, 0); | |
| } else { | |
| ret.set_mask(y / 2, x / 2, 1); | |
| } | |
| } | |
| } | |
| return ret; | |
| } | |
| MaskedImage MaskedImage::upsample(int new_w, int new_h) const { | |
| const auto size = this->size(); | |
| auto ret = MaskedImage(new_w, new_h); | |
| if (!m_global_mask.empty()) ret.init_global_mask_mat(); | |
| for (int y = 0; y < new_h; ++y) { | |
| for (int x = 0; x < new_w; ++x) { | |
| int yy = y * size.height / new_h; | |
| int xx = x * size.width / new_w; | |
| if (is_globally_masked(yy, xx)) { | |
| ret.set_global_mask(y, x, 1); | |
| ret.set_mask(y, x, 1); | |
| } else { | |
| if (!m_global_mask.empty()) ret.set_global_mask(y, x, 0); | |
| if (is_masked(yy, xx)) { | |
| ret.set_mask(y, x, 1); | |
| } else { | |
| auto source_ptr = get_image(yy, xx); | |
| auto target_ptr = ret.get_mutable_image(y, x); | |
| for (int c = 0; c < 3; ++c) | |
| target_ptr[c] = source_ptr[c]; | |
| ret.set_mask(y, x, 0); | |
| } | |
| } | |
| } | |
| } | |
| return ret; | |
| } | |
| MaskedImage MaskedImage::upsample(int new_w, int new_h, const cv::Mat &new_global_mask) const { | |
| auto ret = upsample(new_w, new_h); | |
| ret.set_global_mask_mat(new_global_mask); | |
| return ret; | |
| } | |
| void MaskedImage::compute_image_gradients() { | |
| if (m_image_grad_computed) { | |
| return; | |
| } | |
| const auto size = m_image.size(); | |
| m_image_grady = cv::Mat(size, CV_8UC3); | |
| m_image_gradx = cv::Mat(size, CV_8UC3); | |
| m_image_grady = cv::Scalar::all(0); | |
| m_image_gradx = cv::Scalar::all(0); | |
| for (int i = 1; i < size.height - 1; ++i) { | |
| const auto *ptr = m_image.ptr<unsigned char>(i, 0); | |
| const auto *ptry1 = m_image.ptr<unsigned char>(i + 1, 0); | |
| const auto *ptry2 = m_image.ptr<unsigned char>(i - 1, 0); | |
| const auto *ptrx1 = m_image.ptr<unsigned char>(i, 0) + 3; | |
| const auto *ptrx2 = m_image.ptr<unsigned char>(i, 0) - 3; | |
| auto *mptry = m_image_grady.ptr<unsigned char>(i, 0); | |
| auto *mptrx = m_image_gradx.ptr<unsigned char>(i, 0); | |
| for (int j = 3; j < size.width * 3 - 3; ++j) { | |
| mptry[j] = (ptry1[j] / 2 - ptry2[j] / 2) + 128; | |
| mptrx[j] = (ptrx1[j] / 2 - ptrx2[j] / 2) + 128; | |
| } | |
| } | |
| m_image_grad_computed = true; | |
| } | |
| void MaskedImage::compute_image_gradients() const { | |
| const_cast<MaskedImage *>(this)->compute_image_gradients(); | |
| } | |