Pdf2docx: μ••μΆ• 였λ₯˜ -2

에 λ§Œλ“  2020λ…„ 10μ›” 20일  Β·  9μ½”λ©˜νŠΈ  Β·  좜처: dothinking/pdf2docx

compression error -2 였λ₯˜κ°€ λ°œμƒν–ˆμŠ΅λ‹ˆλ‹€. λˆ„κ΅¬λ“ μ§€ 포인터λ₯Ό 제곡 ν•  수 μžˆλ‹€λ©΄ 쒋을 κ²ƒμž…λ‹ˆλ‹€.

λ¬Έμ œκ°€ μžˆλŠ” PDFλ₯Ό μ²¨λΆ€ν–ˆμŠ΅λ‹ˆλ‹€.
5_KO.pdf

μ—λŸ¬ λ©”μ‹œμ§€:

Processing Pages: 1/28...mupdf: compression error -2
Traceback (most recent call last):
  File "/Users/erikchan/Downloads/convert.py", line 10, in <module>
    parse(pdf_files[i], docx_files[i])
  File "/Library/Frameworks/Python.framework/Versions/3.9/lib/python3.9/site-packages/pdf2docx/main.py", line 31, in parse
    cv.make_docx(indexes, multi_processing)
  File "/Library/Frameworks/Python.framework/Versions/3.9/lib/python3.9/site-packages/pdf2docx/converter.py", line 118, in make_docx
    self._make_docx(page_indexes)
  File "/Library/Frameworks/Python.framework/Versions/3.9/lib/python3.9/site-packages/pdf2docx/converter.py", line 192, in _make_docx
    self.initialize(page).parse().make_page(self.doc_docx)
  File "/Library/Frameworks/Python.framework/Versions/3.9/lib/python3.9/site-packages/pdf2docx/converter.py", line 172, in initialize
    images, paths = self._paths_extractor.extract_paths(page)
  File "/Library/Frameworks/Python.framework/Versions/3.9/lib/python3.9/site-packages/pdf2docx/shape/Path.py", line 61, in extract_paths
    image = largest.to_image(page) if largest.contains_curve else None
  File "/Library/Frameworks/Python.framework/Versions/3.9/lib/python3.9/site-packages/pdf2docx/shape/Path.py", line 140, in to_image
    return ImagesExtractor.clip_page(page, bbox, zoom)
  File "/Library/Frameworks/Python.framework/Versions/3.9/lib/python3.9/site-packages/pdf2docx/image/Image.py", line 60, in clip_page
    return cls.to_raw_dict(image, bbox)
  File "/Library/Frameworks/Python.framework/Versions/3.9/lib/python3.9/site-packages/pdf2docx/image/Image.py", line 50, in to_raw_dict
    'image': image.getPNGData()
  File "/Library/Frameworks/Python.framework/Versions/3.9/lib/python3.9/site-packages/fitz/fitz.py", line 5899, in getPNGData
    barray = self._getImageData(1)
  File "/Library/Frameworks/Python.framework/Versions/3.9/lib/python3.9/site-packages/fitz/fitz.py", line 5868, in _getImageData
    return _fitz.Pixmap__getImageData(self, format)
RuntimeError: compression error -2

bug enhancement

κ°€μž₯ μœ μš©ν•œ λŒ“κΈ€

python-docx κ°€ μžˆλŠ” λ–  μžˆλŠ” 그림이 일반적인 μš”μ²­μΈ 것 κ°™μŠ΅λ‹ˆλ‹€. 곡유λ₯Ό μœ„ν•΄ μ—¬κΈ° λ¬Έμ„œλ₯Ό μ°Έμ‘°ν•˜μ„Έμš”.

# -*- coding: utf-8 -*-

'''
Implement floating image based on python-docx.

- Text wrapping style: BEHIND TEXT <wp:anchor behindDoc="1">
- Picture position: top-left corner of PAGE `<wp:positionH relativeFrom="page">`.

Create a docx sample (Layout | Positions | More Layout Options) and explore the 
source xml (Open as a zip | word | document.xml) to implement other text wrapping
styles and position modes per `CT_Anchor._anchor_xml()`.
'''

from docx.oxml import parse_xml, register_element_cls
from docx.oxml.ns import nsdecls
from docx.oxml.shape import CT_Picture
from docx.oxml.xmlchemy import BaseOxmlElement, OneAndOnlyOne

# refer to docx.oxml.shape.CT_Inline
class CT_Anchor(BaseOxmlElement):
    """
    ``<w:anchor>`` element, container for a floating image.
    """
    extent = OneAndOnlyOne('wp:extent')
    docPr = OneAndOnlyOne('wp:docPr')
    graphic = OneAndOnlyOne('a:graphic')

    <strong i="7">@classmethod</strong>
    def new(cls, cx, cy, shape_id, pic, pos_x, pos_y):
        """
        Return a new ``<wp:anchor>`` element populated with the values passed
        as parameters.
        """
        anchor = parse_xml(cls._anchor_xml(pos_x, pos_y))
        anchor.extent.cx = cx
        anchor.extent.cy = cy
        anchor.docPr.id = shape_id
        anchor.docPr.name = 'Picture %d' % shape_id
        anchor.graphic.graphicData.uri = (
            'http://schemas.openxmlformats.org/drawingml/2006/picture'
        )
        anchor.graphic.graphicData._insert_pic(pic)
        return anchor

    <strong i="8">@classmethod</strong>
    def new_pic_anchor(cls, shape_id, rId, filename, cx, cy, pos_x, pos_y):
        """
        Return a new `wp:anchor` element containing the `pic:pic` element
        specified by the argument values.
        """
        pic_id = 0  # Word doesn't seem to use this, but does not omit it
        pic = CT_Picture.new(pic_id, filename, rId, cx, cy)
        anchor = cls.new(cx, cy, shape_id, pic, pos_x, pos_y)
        anchor.graphic.graphicData._insert_pic(pic)
        return anchor

    <strong i="9">@classmethod</strong>
    def _anchor_xml(cls, pos_x, pos_y):
        return (
            '<wp:anchor distT="0" distB="0" distL="0" distR="0" simplePos="0" relativeHeight="0" \n'
            '           behindDoc="1" locked="0" layoutInCell="1" allowOverlap="1" \n'
            '           %s>\n'
            '  <wp:simplePos x="0" y="0"/>\n'
            '  <wp:positionH relativeFrom="page">\n'
            '    <wp:posOffset>%d</wp:posOffset>\n'
            '  </wp:positionH>\n'
            '  <wp:positionV relativeFrom="page">\n'
            '    <wp:posOffset>%d</wp:posOffset>\n'
            '  </wp:positionV>\n'                    
            '  <wp:extent cx="914400" cy="914400"/>\n'
            '  <wp:wrapNone/>\n'
            '  <wp:docPr id="666" name="unnamed"/>\n'
            '  <wp:cNvGraphicFramePr>\n'
            '    <a:graphicFrameLocks noChangeAspect="1"/>\n'
            '  </wp:cNvGraphicFramePr>\n'
            '  <a:graphic>\n'
            '    <a:graphicData uri="URI not set"/>\n'
            '  </a:graphic>\n'
            '</wp:anchor>' % ( nsdecls('wp', 'a', 'pic', 'r'), int(pos_x), int(pos_y) )
        )


# refer to docx.parts.story.BaseStoryPart.new_pic_inline
def new_pic_anchor(part, image_descriptor, width, height, pos_x, pos_y):
    """Return a newly-created `w:anchor` element.

    The element contains the image specified by *image_descriptor* and is scaled
    based on the values of *width* and *height*.
    """
    rId, image = part.get_or_add_image(image_descriptor)
    cx, cy = image.scaled_dimensions(width, height)
    shape_id, filename = part.next_id, image.filename    
    return CT_Anchor.new_pic_anchor(shape_id, rId, filename, cx, cy, pos_x, pos_y)


# refer to docx.text.run.add_picture
def add_float_picture(p, image_path_or_stream, width=None, height=None, pos_x=0, pos_y=0):
    """Add float picture at fixed position `pos_x` and `pos_y` to the top-left point of page.
    """
    run = p.add_run()
    anchor = new_pic_anchor(run.part, image_path_or_stream, width, height, pos_x, pos_y)
    run._r.add_drawing(anchor)

# refer to docx.oxml.shape.__init__.py
register_element_cls('wp:anchor', CT_Anchor)


if __name__ == '__main__':

    from docx import Document
    from docx.shared import Inches, Pt

    document = Document()

    # add a floating image
    p = document.add_paragraph()
    add_float_picture(p, 'test.png', width=Inches(5.0), pos_x=Pt(20), pos_y=Pt(30))

    # add text
    p.add_run('Hello World'*50)


    document.save('output.docx')

λͺ¨λ“  9 λŒ“κΈ€

이 μΌ€μ΄μŠ€λ₯Ό μ œκ³΅ν•΄ μ£Όμ…”μ„œ κ°μ‚¬ν•©λ‹ˆλ‹€.

λ§Žμ€ 벑터 κ·Έλž˜ν”½, 즉 μ„ , 곑선 및 μ΄λ“€μ˜ μ‘°ν•©κ³Ό 같은 path pdf에 μ‘΄μž¬ν•©λ‹ˆλ‹€. κ·ΈλŸ¬λ‚˜ ν˜„μž¬ 클리핑 κ²½λ‘œλŠ” pdfμ—μ„œ μ΄λŸ¬ν•œ 경둜λ₯Ό μΆ”μΆœν•  λ•Œ 기술적인 문제둜 인해 이 λΌμ΄λΈŒλŸ¬λ¦¬μ—μ„œ λ¬΄μ‹œλ©λ‹ˆλ‹€. 일뢀 κ²½λ‘œκ°€ μž˜λ¦¬μ§€ μ•Šκ³  νŽ˜μ΄μ§€ 외뢀에 μžˆμ–΄ compression error -2 λ¬Έμ œκ°€ λ°œμƒν•©λ‹ˆλ‹€.

κ²Œλ‹€κ°€ 이 pdfλ₯Ό λ³€ν™˜ν•˜λŠ” 데 두 가지 λ¬Έμ œκ°€ 더 μžˆμŠ΅λ‹ˆλ‹€.

  • 경둜 색상이 잘λͺ»λ˜μ—ˆμŠ΅λ‹ˆλ‹€. κ·Όλ³Έ 원인은 ν˜„μž¬ Device Color Space (Gray/RGB/CMYK)만 κ³ λ €λ˜λŠ” 반면 이 pdf μƒ˜ν”Œμ€ Indexed CS , DeviceN CS 와 같은 특수 색상 곡간을 λ”°λ₯Ό 수 μžˆλ‹€λŠ” κ²ƒμž…λ‹ˆλ‹€.

  • 겹친 이미지가 μ œκ±°λ©λ‹ˆλ‹€. python-docx λŠ” λ³€ν™˜λœ docxλ₯Ό μž‘μ„±ν•˜λŠ” 데 μ μš©λ˜μ§€λ§Œ python-docx λŠ” ν˜„μž¬ 뢀동 μš”μ†Œλ₯Ό μ§€μ›ν•˜μ§€ μ•ŠμŠ΅λ‹ˆλ‹€. λ”°λΌμ„œ 뢀동 μ΄λ―Έμ§€λŠ” νƒ€ν˜‘μœΌλ‘œ μ œκ±°λ©λ‹ˆλ‹€.

λ”°λΌμ„œ λΆˆν–‰νžˆλ„ pdf2docx 은(λŠ”) ν˜„μž¬ pdfλ₯Ό λ³€ν™˜ν•  수 μ—†μŠ΅λ‹ˆλ‹€. μ΅œμ†Œν•œ λ‹€μŒκ³Ό 같은 λ…Έλ ₯을 κΈ°μšΈμ—¬μ•Ό ν•©λ‹ˆλ‹€.

  • pdfμ—μ„œ 경둜λ₯Ό μΆ”μΆœν•  λ•Œ 클립 경둜
  • 더 λ§Žμ€ 색 곡간 κ΅¬ν˜„
  • λ–  λ‹€λ‹ˆλŠ” 이미지 μ†Œκ°œ

λͺ…ν™•ν•œ μ„€λͺ…에 λŒ€ν•΄ @dothinking μ—κ²Œ κ°μ‚¬λ“œλ¦½λ‹ˆλ‹€. λ‚˜λŠ” 이 λ„μ„œκ΄€μ΄ μ§€κΈˆλ³΄λ‹€ 더 유λͺ…ν•˜μ§€ μ•Šλ‹€λŠ” 것에 λ†€λžλ‹€. ν˜„μž¬ 버전은 이미 맀우 ν›Œλ₯­ν•˜κ³  λ§Žμ€ μ‚¬λžŒλ“€μ΄ ν˜œνƒμ„ 받을 수 μžˆλ‹€λŠ” 것을 μ•Œκ³  μžˆμŠ΅λ‹ˆλ‹€.

κ·€ν•˜κ°€ λ‚˜μ—΄ν•œ 문제λ₯Ό ν•΄κ²°ν•˜λŠ” 데 도움이 될 수 μžˆλŠ” 방법을 μ•Œλ €μ£Όμ„Έμš”.

@echan00 κ°μ‚¬ν•©λ‹ˆλ‹€.

이 λ¬Έμ œμ— λŒ€ν•œ λͺ‡ 가지 진행 상황:

  • [x] ν”Œλ‘œνŒ… 이미지가 μ§€μ›λ©λ‹ˆλ‹€ .
  • [ ] 클립 경둜 및 색 곡간 -> λ‹€λ₯Έ μ—…μŠ€νŠΈλ¦Ό 라이브러리 PyMuPDF μ—μ„œ 경둜 μΆ”μΆœμ— λŒ€ν•œ μƒˆλ‘œμš΄ κΈ°λŠ₯을 κ²Œμ‹œν–ˆμŠ΅λ‹ˆλ‹€. 쑰사해 보고 이 문제λ₯Ό ν•΄κ²°ν•  수 있기λ₯Ό λ°”λžλ‹ˆλ‹€.

κ·Έ ν›„μ—λŠ” μ–΄λ–€ ν…ŒμŠ€νŠΈλ‚˜ μ œμ•ˆλ„ ν™˜μ˜ν•©λ‹ˆλ‹€.

2020-12-31에 λŒ€ν•œ λŒ“κΈ€: μ΅œμ‹  PyMuPDF 1.18.5λŠ” 이 문제λ₯Ό λΆ€λΆ„μ μœΌλ‘œ ν•΄κ²°ν–ˆμ§€λ§Œ μ™„λ²½ν•˜μ§€λŠ” μ•ŠμŠ΅λ‹ˆλ‹€. 특히 클리핑 κ²½λ‘œκ°€ κ·Έλ ‡μŠ΅λ‹ˆλ‹€.

인라인 μ΄λ―Έμ§€λŠ” python-docx μ—μ„œ μ§€μ›λ˜λ―€λ‘œ ν”Œλ‘œνŒ… 이미지λ₯Ό νƒμƒ‰ν•˜λŠ” λ‹¨κ³„λŠ” λ‹€μŒκ³Ό κ°™μŠ΅λ‹ˆλ‹€.

  • 두 개의 docx νŒŒμΌμ„ λ§Œλ“­λ‹ˆλ‹€. ν•˜λ‚˜λŠ” 인라인 이미지이고 λ‹€λ₯Έ ν•˜λ‚˜λŠ” 뢀동 μ΄λ―Έμ§€μž…λ‹ˆλ‹€(이 경우 behind text λͺ¨λ“œ).
  • 이 두 파일 κ°„μ˜ μ†ŒμŠ€ xml 차이λ₯Ό ν™•μΈν•˜μ‹­μ‹œμ˜€.
  • κ΄€μ°°λœ ꡬ쑰와 인라인 이미지에 λŒ€ν•œ μ½”λ“œλ₯Ό 기반으둜 ν”Œλ‘œνŒ… 이미지 κ΅¬ν˜„

xml ꡬ쑰 결과:

  • 인라인 μ΄λ―Έμ§€λŠ” <w:drawing> μ•„λž˜μ˜ <wp:inline> λ…Έλ“œμž…λ‹ˆλ‹€.
  • ν”Œλ‘œνŒ… μ΄λ―Έμ§€λŠ” <w:drawing> μ•„λž˜μ˜ <wp:anchor> λ…Έλ“œμž…λ‹ˆλ‹€.
  • 인라인 μ΄λ―Έμ§€μ˜ λͺ¨λ“  ν•˜μœ„ λ…Έλ“œ 외에도 뢀동 μ΄λ―Έμ§€μ—λŠ” κ³ μ • μœ„μΉ˜λ₯Ό μ •μ˜ν•˜λŠ” <wp:positionH> 및 <wp:positionV> λ©λ‹ˆλ‹€.

λ”°λΌμ„œ μ•„μ΄λ””μ–΄λŠ” <wp:anchor> λ…Έλ“œλ₯Ό λ§Œλ“  λ‹€μŒ ν•˜μœ„ λ…Έλ“œλ₯Ό μΆ”κ°€ν•˜λŠ” κ²ƒμž…λ‹ˆλ‹€.

  • 인라인 이미지와 λ™μΌν•œ λͺ¨λ“  λ…Έλ“œ
  • <wp:positionH> 및 <wp:positionV>

python-docx κ°€ μžˆλŠ” λ–  μžˆλŠ” 그림이 일반적인 μš”μ²­μΈ 것 κ°™μŠ΅λ‹ˆλ‹€. 곡유λ₯Ό μœ„ν•΄ μ—¬κΈ° λ¬Έμ„œλ₯Ό μ°Έμ‘°ν•˜μ„Έμš”.

# -*- coding: utf-8 -*-

'''
Implement floating image based on python-docx.

- Text wrapping style: BEHIND TEXT <wp:anchor behindDoc="1">
- Picture position: top-left corner of PAGE `<wp:positionH relativeFrom="page">`.

Create a docx sample (Layout | Positions | More Layout Options) and explore the 
source xml (Open as a zip | word | document.xml) to implement other text wrapping
styles and position modes per `CT_Anchor._anchor_xml()`.
'''

from docx.oxml import parse_xml, register_element_cls
from docx.oxml.ns import nsdecls
from docx.oxml.shape import CT_Picture
from docx.oxml.xmlchemy import BaseOxmlElement, OneAndOnlyOne

# refer to docx.oxml.shape.CT_Inline
class CT_Anchor(BaseOxmlElement):
    """
    ``<w:anchor>`` element, container for a floating image.
    """
    extent = OneAndOnlyOne('wp:extent')
    docPr = OneAndOnlyOne('wp:docPr')
    graphic = OneAndOnlyOne('a:graphic')

    <strong i="7">@classmethod</strong>
    def new(cls, cx, cy, shape_id, pic, pos_x, pos_y):
        """
        Return a new ``<wp:anchor>`` element populated with the values passed
        as parameters.
        """
        anchor = parse_xml(cls._anchor_xml(pos_x, pos_y))
        anchor.extent.cx = cx
        anchor.extent.cy = cy
        anchor.docPr.id = shape_id
        anchor.docPr.name = 'Picture %d' % shape_id
        anchor.graphic.graphicData.uri = (
            'http://schemas.openxmlformats.org/drawingml/2006/picture'
        )
        anchor.graphic.graphicData._insert_pic(pic)
        return anchor

    <strong i="8">@classmethod</strong>
    def new_pic_anchor(cls, shape_id, rId, filename, cx, cy, pos_x, pos_y):
        """
        Return a new `wp:anchor` element containing the `pic:pic` element
        specified by the argument values.
        """
        pic_id = 0  # Word doesn't seem to use this, but does not omit it
        pic = CT_Picture.new(pic_id, filename, rId, cx, cy)
        anchor = cls.new(cx, cy, shape_id, pic, pos_x, pos_y)
        anchor.graphic.graphicData._insert_pic(pic)
        return anchor

    <strong i="9">@classmethod</strong>
    def _anchor_xml(cls, pos_x, pos_y):
        return (
            '<wp:anchor distT="0" distB="0" distL="0" distR="0" simplePos="0" relativeHeight="0" \n'
            '           behindDoc="1" locked="0" layoutInCell="1" allowOverlap="1" \n'
            '           %s>\n'
            '  <wp:simplePos x="0" y="0"/>\n'
            '  <wp:positionH relativeFrom="page">\n'
            '    <wp:posOffset>%d</wp:posOffset>\n'
            '  </wp:positionH>\n'
            '  <wp:positionV relativeFrom="page">\n'
            '    <wp:posOffset>%d</wp:posOffset>\n'
            '  </wp:positionV>\n'                    
            '  <wp:extent cx="914400" cy="914400"/>\n'
            '  <wp:wrapNone/>\n'
            '  <wp:docPr id="666" name="unnamed"/>\n'
            '  <wp:cNvGraphicFramePr>\n'
            '    <a:graphicFrameLocks noChangeAspect="1"/>\n'
            '  </wp:cNvGraphicFramePr>\n'
            '  <a:graphic>\n'
            '    <a:graphicData uri="URI not set"/>\n'
            '  </a:graphic>\n'
            '</wp:anchor>' % ( nsdecls('wp', 'a', 'pic', 'r'), int(pos_x), int(pos_y) )
        )


# refer to docx.parts.story.BaseStoryPart.new_pic_inline
def new_pic_anchor(part, image_descriptor, width, height, pos_x, pos_y):
    """Return a newly-created `w:anchor` element.

    The element contains the image specified by *image_descriptor* and is scaled
    based on the values of *width* and *height*.
    """
    rId, image = part.get_or_add_image(image_descriptor)
    cx, cy = image.scaled_dimensions(width, height)
    shape_id, filename = part.next_id, image.filename    
    return CT_Anchor.new_pic_anchor(shape_id, rId, filename, cx, cy, pos_x, pos_y)


# refer to docx.text.run.add_picture
def add_float_picture(p, image_path_or_stream, width=None, height=None, pos_x=0, pos_y=0):
    """Add float picture at fixed position `pos_x` and `pos_y` to the top-left point of page.
    """
    run = p.add_run()
    anchor = new_pic_anchor(run.part, image_path_or_stream, width, height, pos_x, pos_y)
    run._r.add_drawing(anchor)

# refer to docx.oxml.shape.__init__.py
register_element_cls('wp:anchor', CT_Anchor)


if __name__ == '__main__':

    from docx import Document
    from docx.shared import Inches, Pt

    document = Document()

    # add a floating image
    p = document.add_paragraph()
    add_float_picture(p, 'test.png', width=Inches(5.0), pos_x=Pt(20), pos_y=Pt(30))

    # add text
    p.add_run('Hello World'*50)


    document.save('output.docx')

쒋은 @dothinking , λ¬Έμ œκ°€ μ •ν™•νžˆ 무엇인지 μ•Œκ³  μžˆλŠ” 것 κ°™μŠ΅λ‹ˆλ‹€. λ‹€μ–‘ν•œ PDFκ°€ μžˆμŠ΅λ‹ˆλ‹€. μ€€λΉ„κ°€ 되면 ν…ŒμŠ€νŠΈλ₯Ό λ„μ™€λ“œλ¦΄ 수 μžˆμŠ΅λ‹ˆλ‹€.

@dothinking κ·€ν•˜μ˜ μ½”λ“œ μƒ˜ν”Œμ— λŒ€ν•΄

이 ν”„λ‘œμ νŠΈμ— λ„ˆλ¬΄ μ˜€λž«λ™μ•ˆ μ‹œκ°„μ„ β€‹β€‹ν• μ• ν•˜μ§€ λͺ»ν–ˆμŠ΅λ‹ˆλ‹€. 이제 이 문제λ₯Ό λΆ€λΆ„μ μœΌλ‘œ ν•΄κ²°ν•  수 μžˆλŠ” μƒˆ 버전 v0.5.0 μ‚¬μš©ν•  수 μžˆμŠ΅λ‹ˆλ‹€.

  • ν”Œλ‘œνŒ… 이미지가 μ§€μ›λ©λ‹ˆλ‹€.
  • 경둜 μΆ”μΆœμ€ μ—…μŠ€νŠΈλ¦Ό 라이브러리 PyMuPDF μ—μ„œ μ§€μ›λ˜μ§€λ§Œ 클리핑 κ²½λ‘œμ™€ 같은 λ³΅μž‘ν•œ λͺ¨μ–‘μ—λŠ” μ ν•©ν•˜μ§€ μ•ŠμŠ΅λ‹ˆλ‹€.

이 μ΅œμ‹  버전을 μ‚¬μš©ν•˜λ©΄ μƒ˜ν”Œ pdfλ₯Ό μ„±κ³΅μ μœΌλ‘œ λ³€ν™˜ν•  수 μžˆμ§€λ§Œ λ³΅μž‘ν•˜κ³  ν™”λ €ν•œ μŠ€νƒ€μΌλ‘œ 인해 λ³€ν™˜λœ docx 파일의 ν’ˆμ§ˆμ„ 높이렀면 μ—¬μ „νžˆ λ§Žμ€ μž‘μ—…μ΄ ν•„μš”ν•©λ‹ˆλ‹€.

μ™€μš° 이것은 ν›Œλ₯­ν•œ μ—…κ·Έλ ˆμ΄λ“œμž…λ‹ˆλ‹€. @dothinking의 노고에 μ§„μ‹¬μœΌλ‘œ κ°μ‚¬λ“œλ¦½λ‹ˆλ‹€.

이 νŽ˜μ΄μ§€κ°€ 도움이 λ˜μ—ˆλ‚˜μš”?
0 / 5 - 0 λ“±κΈ‰