Just realized I can use LibreOffice Impress to collect my scattered scanned images for mathematical derivations on paper work journal and notebook as well as white board. Compared to LibreOffice Draw, the pages in Impress can be easily shifted and reorganized.

  • A VBA script has been written for inserting, positioning and resizing the images automatically in LibreOffice Impress.

    ' Multiple images can be selected and inserted into Impress in batch, one image
    ' per page.
    Sub InsertImagesFromSelectedFolder()
        ' Get the current document (LibreOffice Draw)
        Dim oDoc As Object
        oDoc = ThisComponent
        
        ' Open the folder and check if it's a directory
        Dim files as Object
        files = SelectMultipleImages()
        
        ' Loop through each file in the folder
        Dim i As Integer
        For i = LBound(files) To UBound(files)
            Dim sFile as String
            sFile = files(i)
            ' Check if the file is an image (you can modify this to include more formats)
            If (InStr(LCase(sFile), ".png") > 0 Or InStr(LCase(sFile), ".jpg") > 0 Or InStr(LCase(sFile), ".jpeg") > 0) Then
                ' Create a new page for each image by inserting into DrawPages
                Dim oPages As Object
                oPages = oDoc.getDrawPages()
                Dim oPage As Object
                oPage = oPages.insertNewByIndex(oPages.getCount())
                oPage.setName(ConvertFromURL(sFile))
        
                ' Create the image shape
                Dim oGraph As Object
                oGraph = oDoc.createInstance("com.sun.star.drawing.GraphicObjectShape")
                oGraph.GraphicURL = sFile
        
                ' Get the image size by callign a Bash script, because they cannot
                ' be acquired from LibreOffice VBA directly.
                CallBashScript("/usr/local/bin/scripts/call_get_img_size.sh", ConvertFromURL(sFile))
                Dim oInputFile as Object
                oInputFile = CreateUnoService("com.sun.star.ucb.SimpleFileAccess")
                Dim oInputStream as Object
                oInputStream = oInputFile.openFileRead("/tmp/libreoffice/img-size.txt")
                Dim sFileContent as String
                sFileContent = ReadInputStream(oInputStream)
                oInputStream.closeInput()
                Dim aSizeParts() as String
                aSizeParts = Split(sFileContent, "x")
        
    	    ' Sclae the image with respect to the page size. N.B. The size unit
    	    ' adopted by LibreOffice is 1/100 mm.
                Dim dpi As Integer
                dpi = 168
                Dim oBitmapSize As New com.sun.star.awt.Size
                oBitmapSize.Width = Val(Trim(aSizeParts(0))) / dpi * 25.4 * 100
                oBitmapSize.Height = Val(Trim(aSizeParts(1))) / dpi * 25.4 * 100
        
                Dim oNewSize As New com.sun.star.awt.Size    'New Image size     
                Dim dImageRatio As Double     'Ratio of the height to width
                Dim dPageRatio As Double      'Ratio of the height to width
        
                dImageRatio = CDbl(oBitmapSize.Height) / CDbl(oBitmapSize.Width)
                dPageRatio = CDbl(oPage.Height) / CDbl(oPage.Width)
        
                ' Compare the ratios to see which is wider, relatively speaking
                If oBitmapSize.Width > oPage.Width OR oBitmapSize.Height > oPage.Height Then
    		If dPageRatio > dImageRatio Then
                        oNewSize.Width  = oPage.Width
                        oNewSize.Height = CLng(CDbl(oPage.Width) * dImageRatio)
    		Else
                        oNewSize.Width  = CLng(CDbl(oPage.Height) / dImageRatio)
                        oNewSize.Height = oPage.Height
    		End If
                Else
    		oNewSize = oBitmapSize
                End If
        
                ' Add the image to the current page.
                oGraph.SetSize(oNewSize)
        
                ' Center the image on the page.
                Dim oPosition as new com.sun.star.awt.Point
                oPosition.X = (oPage.Width - oNewSize.Width)/2
                oPosition.Y = (oPage.Height - oNewSize.Height)/2
                oGraph.SetPosition(oPosition)
        
                oPage.add(oGraph)
            End If
        Next i
    End Sub
        
    ' Function to display the file picker dialog and return the selected folder
    Function SelectMultipleImages() As Object
        Dim oFilePicker As Object
        Dim folderPath As String
        
        ' Create the file picker service
        oFilePicker = CreateUnoService("com.sun.star.ui.dialogs.FilePicker")
        
        ' Set dialog properties
        oFilePicker.setMultiSelectionMode (True)
        oFilePicker.setTitle("Select Folder with Images")
        oFilePicker.execute()
        
        SelectMultipleImages = oFilePicker.getSelectedFiles()
    End Function
        
    ' Call a Bash script. LibreOffice VBA can only start a asynchronous process, so
    ' we need to wait for 1000 ms to let the script finish. Also note the argument
    ' list in the string passed to the Bash script does not need to be escaped.
    Sub CallBashScript(sScriptPath As String, sArgs As String)
        Dim oShell As Object
        
        ' Create a Shell object and execute the command
        oShell = CreateUnoService("com.sun.star.system.SystemShellExecute")
        oShell.execute(sScriptPath, sArgs, 0)  ' The second parameter "0" is to run in the background
        Wait 1000
    End Sub
        
    ' Read the content as a string from an input stream.
    Function ReadInputStream(oInputStream As Object) As String
        Dim sContent As String
        Dim oTextStream As Object
        Dim aBytes() As Byte
        Dim nBytes As Integer
        
        ' Read data from input stream
        oTextStream = CreateUnoService("com.sun.star.io.TextInputStream")
        oTextStream.setInputStream(oInputStream)
        oTextStream.setEncoding("UTF-8") ' Adjust encoding if necessary
        
        ' Read the full content
        sContent = oTextStream.readLine()
        
        ' Return content as a string
        ReadInputStream = sContent
    End Function
    

    The Bash script call_get_img_size.sh is as below. It calls identify to get the image information, from which we write the image size into a temporary file.

    echo $(identify "$1" | gawk '{if(match($0, /([[:digit:]]+)x([[:digit:]]+)/, res) != 0) print res[0];}') > /tmp/libreoffice/img-size.txt
    
  • For eye-protection, the white background of each image is converted to light yellow, i.e. RGB 246, 240, 222, with the command line image editing tool ImageMagick.

    #!/bin/bash
    mogrify -fuzz 10% -fill 'rgb(246,240,222)' -opaque white "$file"
    
  • The handwriting in each inserted image can be further edited with Wacom+Krita.
  • The whole Impress document or its internal bookmarks can be further linked into Org mode, so that they can be categorized and tagged in the Zettelkasten note system.