Teoría

El Reconocimiento Óptico de Caracteres (OCR) es una tecnología utilizada para convertir diferentes tipos de documentos, como imágenes, documentos impresos escaneados, fotografías de texto, archivos PDF o imágenes capturadas con una cámara, en datos editables y buscables.

La Minería de Datos (TM) es el proceso de extraer información útil, patrones o conocimiento de textos no estructurados.

Consta de 3 etapas:
1. Obtener datos: El Reconocimiento Óptico de Caracteres (OCR) es una tecnología que permite convertir imágenes de texto en texto editable. También es conocido como extracción de texto de imágenes.
2. Explorar datos: Representación gráfica o visual de los datos para su interpretación. Los métodos más comunes son el Análisis de Sentimientos, la Nube de Palabras y el Topic Modeling.
3. Análisis predictivo: Son las técnicas y modelos estadísticos para predecir resultados futuros. Los modelos más usados son el Random Forest, redes neuronales y regresiones.

Instalar paquetes y llamar librerías

# Manipulación de datos
library(tidyverse)

# OCR
library(tesseract)

# PNG
library(magick)

# Office (Word)
library(officer)

# PDF
library(pdftools)

# Funciones map
library(purrr)

# Análisis de sentimientos
library(syuzhet)

# Minería de texto
library(tm)

# Nube de palabras
library(wordcloud)

# Paleta de colores
library(RColorBrewer)

De imagen PNG a texto en Word

imagen1 <- image_read("/Users/miguel/Desktop/imagen1.PNG")
texto1 <- ocr(imagen1)
texto1
## [1] "Linear regression with one variable x is also known as univariate linear regression\nor simple linear regression. Simple linear regression is used to predict a single\noutput from a single input. This is an example of supervised learning, which means\nthat the data is labeled, i.e., the output values are known in the training data. Let us\nfit a line through the data using simple linear regression as shown in Fig. 4.1.\n"
doc1 <- read_docx()                       # Crea un documento de Word en blanco
doc1 <- doc1 %>% body_add_par(texto1)     # Pega el texto en el documento
print(doc1, target="texto1.docx")         # Guarda el archivo

De imagen PNG en Español a texto en Word

imagen2 <- image_read("/Users/miguel/Desktop/imagen2.PNG")

# Descargar el lenguaje español para OCR
tesseract_download("spa")
## Training data already exists. Overwriting /Users/miguel/Library/Application Support/tesseract5/tessdata/spa.traineddata
## [1] "/Users/miguel/Library/Application Support/tesseract5/tessdata/spa.traineddata"
# Reconocer texto en español
texto2 <- ocr(imagen2, engine = tesseract("spa"))
texto2
## [1] "Un importante, y quizá controversial, asunto político es el que se refiere al efecto del salario mínimo sobre\nlas tasas de desempleo en diversos grupos de trabajadores. Aunque este problema puede ser estudiado con\ndiversos tipos de datos (corte transversal, series de tiempo o datos de panel), suelen usarse las series de\ntiempo para observar los efectos agregados. En la tabla 1.3 se presenta un ejemplo de una base de datos\nde series de tiempo sobre tasas de desempleo y salarios mínimos.\n"
# Crear documento de Word y pegar el texto
doc2 <- read_docx()                       
doc2 <- doc2 %>% body_add_par(texto2)     
print(doc2, target="texto2.docx")         # Guardar el archivo

De PDF a texto en Word

pdf_eso <- pdf_convert("/Users/miguel/Desktop/eso.pdf") %>% 
  map_chr(ocr)   # Devuelve un solo texto por página
## Converting page 1 to eso_1.png... done!
## Converting page 2 to eso_2.png... done!
doc_pdf <- read_docx()
doc_pdf <- doc_pdf %>% body_add_par(paste(pdf_eso, collapse = "\n\n---- PAGE BREAK ----\n\n"))
print(doc_pdf, target = "pdf_texto.docx")

Análisis de Emociones y sentimientos

# Texto proveniente del OCR (ejemplo: pdf_eso)
texto <- pdf_eso    

# Tokenizar (separa el texto en palabras)
texto_palabras <- get_tokens(texto)

# Análisis de sentimientos en español
emociones <- get_nrc_sentiment(texto_palabras, language = "spanish")

# Ver resultados
head(emociones)
##   anger anticipation disgust fear joy sadness surprise trust negative positive
## 1     0            0       0    0   0       0        0     0        0        0
## 2     0            0       0    0   0       0        0     0        0        0
## 3     0            0       0    0   0       0        0     0        0        0
## 4     0            0       0    0   0       0        0     0        0        0
## 5     0            0       0    0   0       0        0     0        0        0
## 6     0            0       0    0   0       0        0     0        0        0
# Gráfico de emociones básicas (Alegría, Tristeza, Ira, etc.)
barplot(colSums(prop.table(emociones[, 1:8])))

# Puntaje de sentimientos (positivo vs negativo)
sentimientos <- (emociones$negative * -1) + emociones$positive

# Gráfico de sentimientos
simple_plot(sentimientos)

Nube de palabras

palabras <- texto_palabras
palabras <- removeWords(palabras, c(stopwords("spanish")))
wordcloud(words=palabras, min.freq =2, rot.per=0, random.order=FALSE)
## Warning in tm_map.SimpleCorpus(corpus, tm::removePunctuation): transformation
## drops documents
## Warning in tm_map.SimpleCorpus(corpus, function(x) tm::removeWords(x,
## tm::stopwords())): transformation drops documents

LS0tCnRpdGxlOiAiT0NSIgphdXRob3I6ICJNaWd1ZWwgQW5nZWwgTG9wZXoiCmRhdGU6ICIyMDI1LTA4LTIwIgpvdXRwdXQ6IAogIGh0bWxfZG9jdW1lbnQ6CiAgICB0b2M6IHRydWUKICAgIHRvY19mbG9hdDogdHJ1ZQogICAgY29kZV9kb3dubG9hZDogVFJVRQogICAgdGhlbWU6IHNpbXBsZXgKLS0tCiFbXSgvVXNlcnMvbWlndWVsL0Rlc2t0b3AvUEFZQVNPLmdpZikKCgo8c3BhbiBzdHlsZT0iY29sb3I6Ymx1ZTsiPiBUZW9yw61hIDwvc3Bhbj4gIAoKRWwgKipSZWNvbm9jaW1pZW50byDDk3B0aWNvIGRlIENhcmFjdGVyZXMgKE9DUikqKiBlcyB1bmEgdGVjbm9sb2fDrWEgdXRpbGl6YWRhIHBhcmEgY29udmVydGlyIGRpZmVyZW50ZXMgdGlwb3MgZGUgZG9jdW1lbnRvcywgY29tbyBpbcOhZ2VuZXMsIGRvY3VtZW50b3MgaW1wcmVzb3MgZXNjYW5lYWRvcywgZm90b2dyYWbDrWFzIGRlIHRleHRvLCBhcmNoaXZvcyBQREYgbyBpbcOhZ2VuZXMgY2FwdHVyYWRhcyBjb24gdW5hIGPDoW1hcmEsIGVuIGRhdG9zIGVkaXRhYmxlcyB5IGJ1c2NhYmxlcy4KCkxhICoqTWluZXLDrWEgZGUgRGF0b3MgKFRNKSoqIGVzIGVsIHByb2Nlc28gZGUgZXh0cmFlciBpbmZvcm1hY2nDs24gw7p0aWwsIHBhdHJvbmVzIG8gY29ub2NpbWllbnRvIGRlIHRleHRvcyBubyBlc3RydWN0dXJhZG9zLiAgCgpDb25zdGEgZGUgMyBldGFwYXM6ICAKMS4gKipPYnRlbmVyIGRhdG9zOioqIEVsIFJlY29ub2NpbWllbnRvIMOTcHRpY28gZGUgQ2FyYWN0ZXJlcyAoT0NSKSBlcyB1bmEgdGVjbm9sb2fDrWEgcXVlIHBlcm1pdGUgY29udmVydGlyIGltw6FnZW5lcyBkZSB0ZXh0byBlbiB0ZXh0byBlZGl0YWJsZS4gVGFtYmnDqW4gZXMgY29ub2NpZG8gY29tbyBleHRyYWNjacOzbiBkZSB0ZXh0byBkZSBpbcOhZ2VuZXMuICAKMi4gKipFeHBsb3JhciBkYXRvczoqKiBSZXByZXNlbnRhY2nDs24gZ3LDoWZpY2EgbyB2aXN1YWwgZGUgbG9zIGRhdG9zIHBhcmEgc3UgaW50ZXJwcmV0YWNpw7NuLiBMb3MgbcOpdG9kb3MgbcOhcyBjb211bmVzIHNvbiBlbCBBbsOhbGlzaXMgZGUgU2VudGltaWVudG9zLCBsYSBOdWJlIGRlIFBhbGFicmFzIHkgZWwgVG9waWMgTW9kZWxpbmcuICAKMy4gKipBbsOhbGlzaXMgcHJlZGljdGl2bzoqKiBTb24gbGFzIHTDqWNuaWNhcyB5IG1vZGVsb3MgZXN0YWTDrXN0aWNvcyBwYXJhIHByZWRlY2lyIHJlc3VsdGFkb3MgZnV0dXJvcy4gTG9zIG1vZGVsb3MgbcOhcyB1c2Fkb3Mgc29uIGVsIFJhbmRvbSBGb3Jlc3QsIHJlZGVzIG5ldXJvbmFsZXMgeSByZWdyZXNpb25lcy4gIAoKIyA8c3BhbiBzdHlsZT0iY29sb3I6cmVkIj4gSW5zdGFsYXIgcGFxdWV0ZXMgeSBsbGFtYXIgbGlicmVyw61hcyA8L3NwYW4+CmBgYHtyIG1lc3NhZ2U9RkFMU0UsIHdhcm5pbmc9RkFMU0V9CiMgTWFuaXB1bGFjacOzbiBkZSBkYXRvcwpsaWJyYXJ5KHRpZHl2ZXJzZSkKCiMgT0NSCmxpYnJhcnkodGVzc2VyYWN0KQoKIyBQTkcKbGlicmFyeShtYWdpY2spCgojIE9mZmljZSAoV29yZCkKbGlicmFyeShvZmZpY2VyKQoKIyBQREYKbGlicmFyeShwZGZ0b29scykKCiMgRnVuY2lvbmVzIG1hcApsaWJyYXJ5KHB1cnJyKQoKIyBBbsOhbGlzaXMgZGUgc2VudGltaWVudG9zCmxpYnJhcnkoc3l1emhldCkKCiMgTWluZXLDrWEgZGUgdGV4dG8KbGlicmFyeSh0bSkKCiMgTnViZSBkZSBwYWxhYnJhcwpsaWJyYXJ5KHdvcmRjbG91ZCkKCiMgUGFsZXRhIGRlIGNvbG9yZXMKbGlicmFyeShSQ29sb3JCcmV3ZXIpCmBgYAoKIyA8c3BhbiBzdHlsZT0iY29sb3I6cmVkIj4gRGUgaW1hZ2VuIFBORyBhIHRleHRvIGVuIFdvcmQgPC9zcGFuPgpgYGB7cn0KaW1hZ2VuMSA8LSBpbWFnZV9yZWFkKCIvVXNlcnMvbWlndWVsL0Rlc2t0b3AvaW1hZ2VuMS5QTkciKQp0ZXh0bzEgPC0gb2NyKGltYWdlbjEpCnRleHRvMQoKZG9jMSA8LSByZWFkX2RvY3goKSAgICAgICAgICAgICAgICAgICAgICAgIyBDcmVhIHVuIGRvY3VtZW50byBkZSBXb3JkIGVuIGJsYW5jbwpkb2MxIDwtIGRvYzEgJT4lIGJvZHlfYWRkX3Bhcih0ZXh0bzEpICAgICAjIFBlZ2EgZWwgdGV4dG8gZW4gZWwgZG9jdW1lbnRvCnByaW50KGRvYzEsIHRhcmdldD0idGV4dG8xLmRvY3giKSAgICAgICAgICMgR3VhcmRhIGVsIGFyY2hpdm8KYGBgCgojIDxzcGFuIHN0eWxlPSJjb2xvcjpyZWQiPiBEZSBpbWFnZW4gUE5HIGVuIEVzcGHDsW9sIGEgdGV4dG8gZW4gV29yZCA8L3NwYW4+CmBgYHtyfQppbWFnZW4yIDwtIGltYWdlX3JlYWQoIi9Vc2Vycy9taWd1ZWwvRGVza3RvcC9pbWFnZW4yLlBORyIpCgojIERlc2NhcmdhciBlbCBsZW5ndWFqZSBlc3Bhw7FvbCBwYXJhIE9DUgp0ZXNzZXJhY3RfZG93bmxvYWQoInNwYSIpCgojIFJlY29ub2NlciB0ZXh0byBlbiBlc3Bhw7FvbAp0ZXh0bzIgPC0gb2NyKGltYWdlbjIsIGVuZ2luZSA9IHRlc3NlcmFjdCgic3BhIikpCnRleHRvMgoKIyBDcmVhciBkb2N1bWVudG8gZGUgV29yZCB5IHBlZ2FyIGVsIHRleHRvCmRvYzIgPC0gcmVhZF9kb2N4KCkgICAgICAgICAgICAgICAgICAgICAgIApkb2MyIDwtIGRvYzIgJT4lIGJvZHlfYWRkX3Bhcih0ZXh0bzIpICAgICAKcHJpbnQoZG9jMiwgdGFyZ2V0PSJ0ZXh0bzIuZG9jeCIpICAgICAgICAgIyBHdWFyZGFyIGVsIGFyY2hpdm8KYGBgCgojIDxzcGFuIHN0eWxlPSJjb2xvcjpyZWQiPiBEZSBQREYgYSB0ZXh0byBlbiBXb3JkIDwvc3Bhbj4KYGBge3J9CnBkZl9lc28gPC0gcGRmX2NvbnZlcnQoIi9Vc2Vycy9taWd1ZWwvRGVza3RvcC9lc28ucGRmIikgJT4lIAogIG1hcF9jaHIob2NyKSAgICMgRGV2dWVsdmUgdW4gc29sbyB0ZXh0byBwb3IgcMOhZ2luYQoKZG9jX3BkZiA8LSByZWFkX2RvY3goKQpkb2NfcGRmIDwtIGRvY19wZGYgJT4lIGJvZHlfYWRkX3BhcihwYXN0ZShwZGZfZXNvLCBjb2xsYXBzZSA9ICJcblxuLS0tLSBQQUdFIEJSRUFLIC0tLS1cblxuIikpCnByaW50KGRvY19wZGYsIHRhcmdldCA9ICJwZGZfdGV4dG8uZG9jeCIpCmBgYAoKCiMgPHNwYW4gc3R5bGU9ImNvbG9yOnJlZCI+IEFuw6FsaXNpcyBkZSBFbW9jaW9uZXMgeSBzZW50aW1pZW50b3MgPC9zcGFuPgpgYGB7cn0KIyBUZXh0byBwcm92ZW5pZW50ZSBkZWwgT0NSIChlamVtcGxvOiBwZGZfZXNvKQp0ZXh0byA8LSBwZGZfZXNvICAgIAoKIyBUb2tlbml6YXIgKHNlcGFyYSBlbCB0ZXh0byBlbiBwYWxhYnJhcykKdGV4dG9fcGFsYWJyYXMgPC0gZ2V0X3Rva2Vucyh0ZXh0bykKCiMgQW7DoWxpc2lzIGRlIHNlbnRpbWllbnRvcyBlbiBlc3Bhw7FvbAplbW9jaW9uZXMgPC0gZ2V0X25yY19zZW50aW1lbnQodGV4dG9fcGFsYWJyYXMsIGxhbmd1YWdlID0gInNwYW5pc2giKQoKIyBWZXIgcmVzdWx0YWRvcwpoZWFkKGVtb2Npb25lcykKCiMgR3LDoWZpY28gZGUgZW1vY2lvbmVzIGLDoXNpY2FzIChBbGVncsOtYSwgVHJpc3RlemEsIElyYSwgZXRjLikKYmFycGxvdChjb2xTdW1zKHByb3AudGFibGUoZW1vY2lvbmVzWywgMTo4XSkpKQoKIyBQdW50YWplIGRlIHNlbnRpbWllbnRvcyAocG9zaXRpdm8gdnMgbmVnYXRpdm8pCnNlbnRpbWllbnRvcyA8LSAoZW1vY2lvbmVzJG5lZ2F0aXZlICogLTEpICsgZW1vY2lvbmVzJHBvc2l0aXZlCgojIEdyw6FmaWNvIGRlIHNlbnRpbWllbnRvcwpzaW1wbGVfcGxvdChzZW50aW1pZW50b3MpCmBgYAoKIyA8c3BhbiBzdHlsZT0iY29sb3I6cmVkIj4gTnViZSBkZSBwYWxhYnJhcyA8L3NwYW4+CmBgYHtyfQpwYWxhYnJhcyA8LSB0ZXh0b19wYWxhYnJhcwpwYWxhYnJhcyA8LSByZW1vdmVXb3JkcyhwYWxhYnJhcywgYyhzdG9wd29yZHMoInNwYW5pc2giKSkpCndvcmRjbG91ZCh3b3Jkcz1wYWxhYnJhcywgbWluLmZyZXEgPTIsIHJvdC5wZXI9MCwgcmFuZG9tLm9yZGVyPUZBTFNFKQpgYGAKCgoK