diff --git a/.gitignore b/.gitignore index 24444a33..04314df7 100644 --- a/.gitignore +++ b/.gitignore @@ -14,3 +14,8 @@ requal_users.sqlite # gh-pages .quarto wiki/ +tests/test.requal + +# dev stuff +test-iframe.R +wip.R diff --git a/DESCRIPTION b/DESCRIPTION index fd163d1b..ca1d4b33 100644 --- a/DESCRIPTION +++ b/DESCRIPTION @@ -1,6 +1,6 @@ Package: requal Title: Shiny Application for Computer-Assisted Qualitative Data Analysis -Version: 1.1.2.9003 +Version: 1.1.3.9000 Authors@R: c( person(given = "Radim", diff --git a/DEV-NOTES.md b/DEV-NOTES.md index 7b246802..ce34c19d 100644 --- a/DEV-NOTES.md +++ b/DEV-NOTES.md @@ -30,6 +30,7 @@ Instruction for local development of `requal` server version. - [x] add code - [x] merge codes - [x] delete code +- [x] edit code - [x] export codebook #### Categories diff --git a/R/app_server.R b/R/app_server.R index c557b49a..8400cbc0 100644 --- a/R/app_server.R +++ b/R/app_server.R @@ -87,6 +87,16 @@ app_server <- function(input, output, session) { shinyjs::show(selector = ".mfb-component--bl") } }) + + # observe screens + observeEvent(input$analyze_link,{ + updateTabsetPanel(session, "tab_menu", input$analyze_link$tab_menu) + glob$analyze_link <- list( + doc_id = input$analyze_link$doc_id, + segment_start = input$analyze_link$segment_start + ) + }) + # shared mod_download_csv_server("download_csv_ui_1", glob) diff --git a/R/db_logging.R b/R/db_logging.R index 3ea8da6a..d762f5cf 100644 --- a/R/db_logging.R +++ b/R/db_logging.R @@ -80,6 +80,14 @@ log_merge_code_record <- function(con, project_id, from, to, user_id){ data = list(merge_from = from, merge_to = to)) } +log_edit_code_record <- function(con, project_id, changes, user_id){ + log_action(con, + user_id = user_id, + project_id = project_id, + action = "Edit code", + data = changes) +} + log_add_segment_record <- function(con, project_id, segment, user_id){ log_action(con, user_id = user_id, @@ -206,4 +214,5 @@ log_change_user_permission <- function(con, project_id, permission_data, user_id project_id, action = "Change user permission", data = permission_data) -} \ No newline at end of file +} + diff --git a/R/db_startup.R b/R/db_startup.R index 60e34195..4ace6e71 100644 --- a/R/db_startup.R +++ b/R/db_startup.R @@ -1,4 +1,4 @@ -utils::globalVariables(c("sql")) +utils::globalVariables(c("sql", "is_new_quickcode")) db_call <- c( @@ -555,11 +555,12 @@ add_cases_record <- function(pool, project_id, case_df, user_id) { } add_codes_record <- function(pool, project_id, codes_df, user_id) { + res <- DBI::dbWriteTable(pool, "codes", codes_df, append = TRUE, row.names = FALSE) if (res) { written_code_id <- dplyr::tbl(pool, "codes") %>% dplyr::filter(.data$code_name == !!codes_df$code_name, - .data$project_id == !!as.numeric(project_id), + .data$project_id == !!as.integer(project_id), .data$user_id == !!user_id) %>% dplyr::pull(code_id) @@ -574,6 +575,36 @@ add_codes_record <- function(pool, project_id, codes_df, user_id) { } } +add_quickcode_record <- function(pool, project_id, codes_df, user_id) { + + # Make sure column exists to identify new quickcode + db_helper_column(pool, "codes", "is_new_quickcode", "add") + # temporarily write into DB with original code_id + codes_df$is_new_quickcode <- 1 + + res <- DBI::dbWriteTable(pool, "codes", codes_df, append = TRUE, row.names = FALSE) + if (res) { + written_code_id <- dplyr::tbl(pool, "codes") %>% + dplyr::filter(.data$project_id == !!as.integer(project_id), + .data$user_id == !!as.integer(user_id), + is_new_quickcode == 1) %>% + dplyr::pull(code_id) + # remove helper column from DB + db_helper_column(pool, "codes", "is_new_quickcode", "drop") + # just a check we are getting the latest id + written_code_id <- written_code_id[written_code_id == max(written_code_id)] + log_add_code_record(pool, project_id, codes_df %>% + dplyr::mutate( + is_new_quickcode = NULL, + code_id = written_code_id), + user_id + ) + return(written_code_id) + } else { + warning("code not added") + } +} + add_case_doc_record <- function(pool, project_id, case_doc_df, user_id) { res <- DBI::dbWriteTable(pool, "cases_documents_map", case_doc_df, append = TRUE, row.names = FALSE) if (res) { @@ -627,4 +658,5 @@ make_globals <- quote({ existing_projects <- data.frame() } } -}) \ No newline at end of file +}) + diff --git a/R/import_rqda.R b/R/import_rqda.R index 71956c39..0ee82615 100644 --- a/R/import_rqda.R +++ b/R/import_rqda.R @@ -41,8 +41,8 @@ rql_import_rqda <- function(rqda_file, requal_file){ RSQLite::SQLite(), dbname = requal_file ) - - message("Loading data from RQDA") + + rql_message("Loading data from RQDA") # Load Data from RQDA project_df <- dplyr::tbl(rqda_con, "project") %>% dplyr::collect() %>% @@ -117,7 +117,7 @@ rql_import_rqda <- function(rqda_file, requal_file){ dplyr::filter(!is.na(code_id)) # Create requal schema - message("Creating Requal scheme") + rql_message("Creating Requal scheme") create_db_schema(requal_pool) # Import to requal @@ -129,7 +129,7 @@ rql_import_rqda <- function(rqda_file, requal_file){ dplyr::pull(project_id) %>% utils::tail(1) - message("Importing documents") + rql_message("Importing documents") documents_df <- rqda_documents %>% dplyr::mutate(project_id = requal_project_id) purrr::walk(seq_len(nrow(documents_df)), function(x) { @@ -137,7 +137,7 @@ rql_import_rqda <- function(rqda_file, requal_file){ user_id = USER_ID) }) - message("Importing cases") + rql_message("Importing cases") cases_df <- rqda_cases %>% dplyr::mutate(project_id = requal_project_id) purrr::walk(seq_len(nrow(cases_df)), function(x) { @@ -145,7 +145,7 @@ rql_import_rqda <- function(rqda_file, requal_file){ user_id = USER_ID) }) - message("Importing case document map") + rql_message("Importing case document map") case_doc_map <- rqda_case_doc_map %>% dplyr::mutate(project_id = requal_project_id) purrr::walk(seq_len(nrow(case_doc_map)), function(x) { @@ -157,7 +157,7 @@ rql_import_rqda <- function(rqda_file, requal_file){ dplyr::filter(is.na(code_color)) %>% nrow() - message("Importing codes") + rql_message("Importing codes") codes_df <- rqda_codes %>% dplyr::mutate(project_id = requal_project_id, code_color = ifelse(is.na(code_color), @@ -168,7 +168,7 @@ rql_import_rqda <- function(rqda_file, requal_file){ user_id = USER_ID) }) - message("Importing categories") + rql_message("Importing categories") categories_df <- rqda_categories %>% dplyr::mutate(project_id = requal_project_id) purrr::walk(seq_len(nrow(categories_df)), function(x) { @@ -177,7 +177,7 @@ rql_import_rqda <- function(rqda_file, requal_file){ user_id = USER_ID) }) - message("Importing category code mapping") + rql_message("Importing category code mapping") category_code_map <- rqda_category_code_map %>% dplyr::mutate(project_id = requal_project_id) purrr::walk(seq_len(nrow(category_code_map)), function(x) { @@ -186,7 +186,7 @@ rql_import_rqda <- function(rqda_file, requal_file){ user_id = USER_ID) }) - message("Importing segments") + rql_message("Importing segments") segments_df <- rqda_segments %>% dplyr::mutate(project_id = requal_project_id, segment_text = purrr::pmap_chr( @@ -204,7 +204,7 @@ rql_import_rqda <- function(rqda_file, requal_file){ user_id = USER_ID) }) - message("Importing memos") + rql_message("Importing memos") if(!all(is.na(rqda_segments$memo))){ DBI::dbWriteTable(requal_pool, "memos", memos_df %>% dplyr::select(memo_id, text), append = TRUE, row.names = FALSE) diff --git a/R/mod_analysis.R b/R/mod_analysis.R index fb481768..cc724818 100644 --- a/R/mod_analysis.R +++ b/R/mod_analysis.R @@ -200,16 +200,20 @@ mod_rql_button_server( if (nrow(loc$segments_df) > 0) { loc$segments_taglist <- purrr::pmap( list( + loc$segments_df$segment_start, loc$segments_df$segment_text, + loc$segments_df$doc_id, loc$segments_df$doc_name, loc$segments_df$code_name, loc$segments_df$code_color ), ~ format_segments( - segment_text = ..1, - segment_document = ..2, - segment_code = ..3, - segment_color = ..4 + segment_start = ..1, + segment_text = ..2, + segment_document_id = ..3, + segment_document_name = ..4, + segment_code = ..5, + segment_color = ..6 ) ) } diff --git a/R/mod_analysis_utils_analysis.R b/R/mod_analysis_utils_analysis.R index 7de442cd..3bc2c335 100644 --- a/R/mod_analysis_utils_analysis.R +++ b/R/mod_analysis_utils_analysis.R @@ -50,7 +50,7 @@ load_segments_analysis <- function(pool, -format_segments <- function(segment_text, segment_document, segment_code, segment_color) { +format_segments <- function(segment_start, segment_text, segment_document_id, segment_document_name, segment_code, segment_color) { tags$div( @@ -59,8 +59,11 @@ format_segments <- function(segment_text, segment_document, segment_code, segmen tags$blockquote(class = "quote", style = paste0("border-left: 5px solid ", segment_color, "; margin-bottom: 0px !important;")), tags$div( - segment_document %>% - tags$div(class = "segment_badge"), + tags$div(class = "segment_badge", + actionLink(paste0("segment_start-", segment_start), label = segment_document_name, + onclick = paste0("Shiny.setInputValue('analyze_link', {tab_menu: 'Annotate', doc_id: ", segment_document_id,", segment_start: ", segment_start, "}, {priority: 'event'});") + ) + ), segment_code %>% tags$div(class = "segment_badge", style = paste0("background-color: ", segment_color, " !important;")), diff --git a/R/mod_browser.R b/R/mod_browser.R index b0e12c4b..fb5d9582 100644 --- a/R/mod_browser.R +++ b/R/mod_browser.R @@ -170,7 +170,7 @@ mod_browser_server <- function(id, glob){ tibble::tibble( position_start = 0, position_type = "segment_start", - tag_start = "

" + tag_start = "

" ), # content diff --git a/R/mod_categories.R b/R/mod_categories.R index d9998541..5d42855c 100644 --- a/R/mod_categories.R +++ b/R/mod_categories.R @@ -95,7 +95,8 @@ mod_categories_server <- function(id, glob) { }) # Relist categories on codebook changes --------------- - observeEvent(glob$codebook, { + observeEvent(c(glob$codebook, + glob$codebook_observer), { output$categories_ui <- renderUI({ render_categories( id = id, diff --git a/R/mod_codebook.R b/R/mod_codebook.R index 2a3810b4..8d3ccf37 100644 --- a/R/mod_codebook.R +++ b/R/mod_codebook.R @@ -23,10 +23,14 @@ mod_codebook_ui <- function(id) { label = "Create code", icon = "plus" ), + mod_rql_button_ui(ns("code_edit_ui"), + label = "Edit codes", + icon = "edit" + ), mod_rql_button_ui(ns("code_merge_ui"), label = "Merge codes", icon = "compress" - ), + ), mod_rql_button_ui(ns("code_delete_ui"), label = "Delete code", icon = "minus" @@ -65,8 +69,10 @@ mod_codebook_server <- function(id, glob) { observeEvent(c( glob$active_project, input$code_add, + input$code_edit_btn, input$code_merge, - input$code_del_btn + input$code_del_btn, + glob$codebook_observer ), { #---Create code UI -------------- mod_rql_button_server( @@ -84,6 +90,14 @@ mod_codebook_server <- function(id, glob) { glob, permission = "codebook_modify" ) + #---Edit code UI -------------- + mod_rql_button_server( + id = "code_edit_ui", + custom_title = "Edit code", + custom_tagList = edit_code_UI(ns, glob$pool, glob$active_project, glob$user), + glob, + permission = "codebook_modify" + ) #---Delete code UI -------------- mod_rql_button_server( id = "code_delete_ui", @@ -148,6 +162,75 @@ mod_codebook_server <- function(id, glob) { } }) + #---Edit existing code------------------------------------- + # To edit a code, first observe which code is being edited + observeEvent(input$code_to_edit, { + req(input$code_to_edit) + updateTextInput( + session = session, + "edit_code_name", + value = glob$codebook %>% + dplyr::filter(code_id == input$code_to_edit) %>% + dplyr::pull(code_name) + ) + updateTextAreaInput( + session = session, + "edit_code_desc", + value = glob$codebook %>% + dplyr::filter(code_id == input$code_to_edit) %>% + dplyr::pull(code_description) + ) + colourpicker::updateColourInput( + session = session, + "edit_color_pick", + value = glob$codebook %>% + dplyr::filter(code_id == input$code_to_edit) %>% + dplyr::pull(code_color) + ) + }) + # Execute code edit + observeEvent(input$code_edit_btn, { + req(input$code_to_edit) + + # check if code name is unique + code_names <- list_db_codes( + pool = glob$pool, + project_id = glob$active_project, + user = glob$user + ) %>% + dplyr::filter(code_id != input$code_to_edit) %>% # exclude original name from comparison + dplyr::pull(code_name) + + # code must have a name that does not exist yet (unless it stays the same as original) + if (isTruthy(input$edit_code_name) && !input$edit_code_name %in% code_names) { + + # edit code + edit_db_codes( + pool = glob$pool, + active_project = glob$active_project, + user_id = glob$user$user_id, + edit_code_id = input$code_to_edit, + edit_code_name = input$edit_code_name, + edit_code_description = input$edit_code_desc, + edit_code_color = paste0( + "rgb(", + paste( + as.vector( + grDevices::col2rgb( + input$edit_color_pick + ) + ), + collapse = ", " + ), + ")" + ) + ) + } else { + warn_user("Code names must be unique and non-empty.") + } + + }) + #---Delete existing code------------------------------------- observeEvent(input$code_del_btn, { req(input$code_to_del) diff --git a/R/mod_codebook_utils_codebook.R b/R/mod_codebook_utils_codebook.R index 724a1aa8..0d7751b8 100644 --- a/R/mod_codebook_utils_codebook.R +++ b/R/mod_codebook_utils_codebook.R @@ -63,6 +63,53 @@ merge_code_UI <- function(ns, pool, project, user) { } +edit_code_UI <- function(ns, pool, project, user) { + + req(user$data) + + codes <- list_db_codes( + pool, + project_id = project, + user = user + ) + + if(user$data$codebook_other_modify == 0){ + codes <- codes %>% + dplyr::filter(user_id == !!user$user_id) + } + tags$div( + selectizeInput( + ns("code_to_edit"), + label = "Select code to edit", + choices = c("", stats::setNames(codes$code_id, codes$code_name)), + selected = NULL, + multiple = FALSE, + options = list( + closeAfterSelect = "true" + ) + ), + textInput( + ns("edit_code_name"), + label = "Code name" + ) %>% tagAppendAttributes(class = "required"), + textAreaInput( + ns("edit_code_desc"), + label = "Code description" + ), + colourpicker::colourInput( + ns("edit_color_pick"), + label = "Highlight", + value = "white", + showColour = "background", + closeOnClick = TRUE + ), + actionButton(ns("code_edit_btn"), + label = "Edit", + class = "btn-warning" + ) + ) %>% tagAppendAttributes(style = "text-align: left") +} + delete_code_UI <- function(ns, pool, project, user) { req(user$data) @@ -83,7 +130,7 @@ delete_code_UI <- function(ns, pool, project, user) { label = "Select codes to delete", choices = stats::setNames(codes$code_id, codes$code_name), selected = NULL, - multiple = TRUE, + multiple = FALSE, options = list( closeAfterSelect = "true" ) @@ -233,7 +280,6 @@ render_codes <- function(pool, active_project, user) { project_id = active_project, user = user ) - if (nrow(project_codes) == 0) { "No codes have been created." } else { @@ -293,3 +339,29 @@ get_codebook_export_table <- function(glob){ dplyr::group_by(code_id, code_name, code_description) %>% dplyr::summarise(categories = paste0(category_title, collapse = " | ")) } + +# Edit codes ------ + +edit_db_codes <- function(pool, + active_project, + user_id, + edit_code_id, + edit_code_name, + edit_code_description, + edit_code_color) { + + update_code_sql <- glue::glue_sql("UPDATE codes + SET code_name = {edit_code_name}, code_description = {edit_code_description}, code_color = {edit_code_color} + WHERE code_id = {edit_code_id}", .con = pool) + DBI::dbExecute(pool, update_code_sql) + + log_edit_code_record(pool, project_id = active_project, + changes = list( + code_id = edit_code_id, + code_name = edit_code_name, + code_color = edit_code_color, + code_description = edit_code_description), + user_id) + + rql_message(paste("Code", edit_code_name, "was updated.")) +} diff --git a/R/mod_doc_manager.R b/R/mod_doc_manager.R index f81c0b17..cbc8caf8 100644 --- a/R/mod_doc_manager.R +++ b/R/mod_doc_manager.R @@ -216,8 +216,8 @@ mod_doc_manager_server <- function(id, glob) { ) }, error = function(e) { - message("Error in adding document: ", e$message) - "failed" + rql_message(paste0("Error in adding document: ", e$message)) + return("failed") } ) diff --git a/R/mod_document_code.R b/R/mod_document_code.R index 84a85570..261ec1b6 100644 --- a/R/mod_document_code.R +++ b/R/mod_document_code.R @@ -9,43 +9,93 @@ #' @importFrom shiny NS tagList mod_document_code_ui <- function(id) { ns <- NS(id) -tagList( - fluidRow(class = "module_tools", style = "width: 100%", - div(style="display: flex; justify-content: space-between; width: 100%", - selectInput(ns("doc_selector"), - label = "Select a document to code", - choices = "", selected = "" - ), - div(style = "display: flex; align-items: center;", - actionButton(ns("doc_refresh"), - label = "", - icon = icon("sync") - ) %>% tagAppendAttributes(title = "Reload document") - ) - ) -), -fluidRow( - column( - width = 10, - htmlOutput(ns("focal_text")) %>% tagAppendAttributes(class = "scrollable80"), - textOutput(ns("captured_range")), - verbatimTextOutput(ns("printLabel")) + + fluidPage( + tags$head( + tags$script(src = "www/split.min.js"), + tags$script(src = "www/highlight_style.js"), + tags$script(src = "www/document_code_js.js"), + tags$script(HTML(" + document.addEventListener('DOMContentLoaded', (event) => { + Split(['#split-1', '#split-2'], { + sizes: [80, 20], + minSize: [100, 100] + }); + }); +")) ), - column( - width = 2, - tags$b("Codes"), - br(), - actionButton(ns("remove_codes"), - "Remove code", - class = "btn-danger", - width = "100%" + fluidRow( + column( + width = 5, + selectInput( + ns("doc_selector"), + label = "Select a document to code", + choices = "", + selected = "" + ) ), - br(), br(), - uiOutput(ns("code_list")) - ) %>% tagAppendAttributes(class = "scrollable90") - ), - tags$script( - src = "www/document_code_js.js" + column( + width = 5, + div(style = "float: right;", + actionButton( + ns("toggle_style"), + label = "", + icon = icon("highlighter") + ) %>% tagAppendAttributes(title = "Highlight style"), + actionButton( + ns("doc_refresh"), + label = "", + icon = icon("sync") + ) %>% tagAppendAttributes(title = "Reload document") + )), + column( + width = 2, + div(style = "text-align: right;", + actionButton( + ns("code_columns"), + label = "", + icon = icon("table-columns") + ) %>% tagAppendAttributes(title = "Code columns"), + br(), "Selection:", br(), + textOutput(ns("captured_range")) + ) + ) + ), + fluidRow( + style = "height: 90%", + tags$div( + style = "display: flex;", + class = "split", + tags$div( + id = "split-1", + style = "flex-grow: 1; flex-shrink: 1; overflow: auto;", + htmlOutput(ns("focal_text")) %>% tagAppendAttributes(class = "scrollable80") + ), + tags$div( + id = "split-2", + style = "flex-grow: 1; flex-shrink: 1; overflow: auto;", + tags$b("Codes"), + br(), + actionButton( + ns("remove_codes"), + "Remove code", + class = "btn-danger", + width = "100%" + ), + tags$div( + style = "height: calc(1.5em + .75rem + 10px); display: flex; align-items: center; margin-top: 0.5em; margin-bottom: 0.5em;", + tags$div( + style = "flex-grow: 1; height: calc(1.5em + .75rem + 10px);", + tags$iframe( + src = "www/quickcode.html", + style = "height: calc(1.5em + .75rem + 10px); width: 100%; border: none;" + ) + ), + actionButton(ns("quickcode_btn"), "Quick tag", icon = icon("bolt-lightning"), style = "height: calc(1.5em + .75rem + 10px); margin-left: 0px;") + ), + uiOutput(ns("code_list")) %>% tagAppendAttributes(class = "scrollable80") + ) + ) ) ) } @@ -56,139 +106,241 @@ fluidRow( mod_document_code_server <- function(id, glob) { moduleServer(id, function(input, output, session) { ns <- session$ns - loc <- reactiveValues() - - # Selection of documents to code ------------------------------------------ - loc$doc_choices <- NULL + loc$highlight <- "background" + loc$code <- NULL + observeEvent(req(glob$active_project), { + loc$codes_menu_observer <- 0 + loc$code_action_observer <- 0 + loc$text_observer <- 0 + loc$scroll <- 0 + }) + + # Observers - definitions ---- + ## Observe click on coded text ---- + observeEvent(input$clicked_title, { + showNotification(input$clicked_title) + }) + ## Observe choice of highlight style ---- + observeEvent(input$toggle_style, { + loc$highlight <- ifelse(loc$highlight == "underline", "background", "underline") + # Send a message to the client to toggle the style + session$sendCustomMessage("toggleStyle", message = loc$highlight) + }) - # Refresh list of documents when documents are added/removed -------- + ## Observe changes in documents observeEvent(glob$documents, { if (isTruthy(glob$active_project)) { - if(glob$user$data$data_other_view == 1){ + if (glob$user$data$data_other_view == 1) { updateSelectInput( session = session, "doc_selector", choices = c("", glob$documents) - ) - }else{ - visible_docs <- read_visible_docs(glob$pool, glob$active_project, - glob$user$user_id) + ) + } else { + visible_docs <- read_visible_docs( + glob$pool, glob$active_project, + glob$user$user_id + ) updateSelectInput( - session = session, - "doc_selector", + session = session, + "doc_selector", choices = c("", visible_docs) ) } - } }) - - # Displayed text ---------------------------------------------------------- - loc$text <- "" - - observeEvent(c(input$doc_selector, input$doc_refresh), { - req(isTruthy(input$doc_selector)) - loc$text <- load_doc_to_display( - glob$pool, - glob$active_project, - user = glob$user, - input$doc_selector, - loc$code_df$active_codebook, - ns = NS(id) - ) - glob$segments_observer <- glob$segments_observer + 1 - + ## Doc sel or refresh ---- + # Update loc$text when input$doc_selector or input$doc_refresh changes + observeEvent(c(input$doc_selector, + input$doc_refresh), { + req(input$doc_selector) + loc$codes_menu_observer <- loc$codes_menu_observer + 1 # must run first + loc$text_observer <- loc$text_observer + 1 }) - # Render selected text - output$focal_text <- renderText({ - loc$text - }) - - - - # List out available codes ------------------------------------------------ - output$code_list <- renderUI({ - if (isTruthy(glob$active_project)) { - if (isTruthy(glob$codebook)) { - loc$code_df$active_codebook <- glob$codebook - } else { - loc$code_df$active_codebook <- list_db_codes( - glob$pool, - glob$active_project - ) - } + ## Observe refresh ---- + # Update loc$codes_menu when input$doc_refresh or glob$codebook changes + observeEvent(input$doc_refresh, { + loc$codes_menu_observer <- loc$codes_menu_observer + 1 + }) - if (isTruthy(req(input$doc_selector))) { - loc$text <- load_doc_to_display( - glob$pool, - glob$active_project, - user = glob$user, - input$doc_selector, - loc$code_df$active_codebook, - ns = NS(id) - ) - } + ## Code columns observer ----- + observeEvent(input$code_columns, { + shinyjs::toggleClass(id = "codes_menu", "two_columns", asis = TRUE) + }) - purrr::pmap( + ## Codes menu observer ---- + observeEvent(req(loc$codes_menu_observer), { + req(loc$codes_menu_observer > 0) # ignore init value + loc$codebook <- list_db_codes( + glob$pool, + glob$active_project, + user = glob$user + ) + + code_labels <- purrr::pmap( list( - loc$code_df$active_codebook$code_id, - loc$code_df$active_codebook$code_name, - loc$code_df$active_codebook$code_color, - loc$code_df$active_codebook$code_description + loc$codebook$code_id, + loc$codebook$code_name, + loc$codebook$code_color, + loc$codebook$code_description ), ~ generate_coding_tools( ns = ns, code_id = ..1, code_name = ..2, code_color = ..3, - code_desc = ..4) - ) + code_desc = ..4 + )) + + loc$codes_menu <- sortable::rank_list( + input_id = "codes_menu", + css_id = "codes_menu", + labels = code_labels + ) + }) + ## Observe Analyze screen ---- + # Listen to message from Analyze screen + observeEvent(glob$analyze_link, { + updateSelectInput(session = session, "doc_selector", choices = c("", glob$documents), selected = glob$analyze_link$doc_id) + loc$codes_menu_observer <- loc$codes_menu_observer + 1 + loc$text_observer <- loc$text_observer + 1 + loc$scroll <- loc$scroll + 1 + }) + observeEvent(loc$scroll , { + req(isTruthy(loc$scroll)) + session$sendCustomMessage(type = 'scrollToSegment', message = glob$analyze_link$segment_start) + }) - } else { - "" - } + # Render text and codes ---- + output$focal_text <- renderText({ + req(isTruthy(loc$text)) + loc$text + }) + output$code_list <- renderUI({ + req(isTruthy(loc$codes_menu)) + loc$codes_menu }) - # Coding tools ------------------------------------------------------------ - observeEvent(input$selected_code, { - req(input$selected_code, input$tag_position) + # Load text observer ---- + observeEvent(req(loc$text_observer), { + req(loc$text_observer > 0) # ignore init value + loc$text <- load_doc_to_display( + glob$pool, + glob$active_project, + user = glob$user, + ifelse(isTruthy(input$doc_selector), input$doc_selector, glob$analyze_link$doc_id), + loc$codebook, + highlight = loc$highlight, + ns = NS(id) + ) + }) - startOff <- parse_tag_pos(input$tag_position, "start") - endOff <- parse_tag_pos(input$tag_position, "end") + # Coding tools ------------------------------------------------------------ + observeEvent(req(input$selected_code), { + # We need a document and selection positions + req(input$doc_selector) + req(input$tag_position) + # Register code for which action is executed + loc$code <- input$selected_code + # Call code action observer + loc$code_action_observer <- loc$code_action_observer + 1 + }) + ## Codes action observer ---- + # Write code to DB when observer is updated + observeEvent(req(loc$code_action_observer), { + req(loc$code_action_observer > 0) # ignore init value + req(loc$code) + # To execute, we need a document and a selection + startOff <- parse_tag_pos(req(input$tag_position), "start") + endOff <- parse_tag_pos(req(input$tag_position), "end") - if (endOff - startOff > 1 & endOff > startOff) { + if (endOff >= startOff) { write_segment_db( - glob$pool, - glob$active_project, - user_id = glob$user$user_id, - doc_id = input$doc_selector, - code_id = input$selected_code, - startOff, - endOff + glob$pool, + glob$active_project, + user_id = glob$user$user_id, + doc_id = input$doc_selector, + code_id = loc$code, + startOff, + endOff ) + # TODO JS implementation of coding + # code_info <- glob$codebook |> + # dplyr::filter(code_id == input$selected_code) - loc$text <- load_doc_to_display( - glob$pool, - glob$active_project, - user = glob$user, - input$doc_selector, - loc$code_df$active_codebook, - ns = NS(id) - ) + # session$sendCustomMessage(type = 'wrapTextWithBold', message = list( + # startOffset = startOff-1, #convert to javascript + # endOffset = endOff, + # newId = input$selected_code, + # color = code_info$code_color, + # title = code_info$code_name + # )) + + loc$text_observer <- loc$text_observer + 1 glob$segments_observer <- glob$segments_observer + 1 } }) + + # Quick code tools ---- + observeEvent(input$quickcode_btn, { + + # We need a document and selection positions + req(input$doc_selector) + req(input$tag_position) + if (parse_tag_pos(input$tag_position, "start") < parse_tag_pos(input$tag_position, "end")) { + session$sendCustomMessage(type = "getIframeContent", message = list()) + session$sendCustomMessage(type = 'refreshIframe', message = list()) + } else { + rql_message("Missing selected text segment.") + } + + }) + # After quickcode button is pressed, we wait for the quickcode value + observeEvent(req(input$quickcode), { + # check if code name is unique + if (input$quickcode %in% glob$codebook$code_name) { + warn_user("Code names must be unique and non-empty.") + } else { + codes_input_df <- data.frame( + project_id = glob$active_project, + code_name = input$quickcode, + code_description = "", + code_color = "rgb(255,255,0)", + user_id = glob$user$user_id + ) + # Add new quickcode to codes database + # and register the new code_id + loc$code <- add_quickcode_record( + pool = glob$pool, + project_id = glob$active_project, + codes_df = codes_input_df, + user_id = glob$user$user_id + ) + # Refresh codes menu + loc$codes_menu_observer <- loc$codes_menu_observer + 1 + # Execute coding action + loc$code_action_observer <- loc$code_action_observer + 1 + # Refresh text + loc$text_observer <- loc$text_observer + 1 + # Notify codebook screen about new quick code + glob$codebook_observer <- ifelse( + !isTruthy(glob$codebook_observer), + 0, glob$codebook_observer + 1) + # Notify user + rql_message(paste(input$quickcode,"added to codebook.")) + } + }) # Segment removal ---------- observeEvent(input$remove_codes, { req(glob$active_project) - req(isTruthy(input$doc_selector)) - - if(glob$user$data$annotation_other_modify == 0){ + req(input$doc_selector) + + if (glob$user$data$annotation_other_modify == 0) { loc$marked_segments_df <- load_segment_codes_db( - glob$pool, + glob$pool, glob$active_project, user_id = glob$user$user_id, active_doc = input$doc_selector, @@ -196,10 +348,10 @@ mod_document_code_server <- function(id, glob) { input$tag_position, "start" ) - ) - }else{ + ) + } else { loc$marked_segments_df <- load_segment_codes_db( - glob$pool, + glob$pool, glob$active_project, user_id = NULL, active_doc = input$doc_selector, @@ -209,33 +361,25 @@ mod_document_code_server <- function(id, glob) { ) ) } - if (nrow(loc$marked_segments_df) == 0) { NULL } else if (nrow(loc$marked_segments_df) == 1) { delete_segment_codes_db( - glob$pool, - glob$active_project, - user_id = glob$user$user_id, - doc_id = input$doc_selector, - segment_id = loc$marked_segments_df$segment_id - ) - - loc$text <- load_doc_to_display( - glob$pool, - glob$active_project, - user = glob$user, - input$doc_selector, - loc$code_df$active_codebook, - ns = NS(id) + glob$pool, + glob$active_project, + user_id = glob$user$user_id, + doc_id = input$doc_selector, + segment_id = loc$marked_segments_df$segment_id ) + # Refresh text + loc$text_observer <- loc$text_observer + 1 + # Notify analysis screen glob$segments_observer <- glob$segments_observer + 1 - } else { + # Obtain additional input if multiple segments are to be removed showModal( modalDialog( - checkboxGroupInput(ns("codes_to_remove"), label = "", choiceValues = loc$marked_segments_df$segment_id, @@ -258,35 +402,37 @@ mod_document_code_server <- function(id, glob) { } }) + # Multiple segments removal ---- observeEvent(input$remove_codes_select, { if (isTruthy(input$codes_to_remove)) { delete_segment_codes_db( - glob$pool, - glob$active_project, - user_id = glob$user$user_id, - doc_id = input$doc_selector, - segment_id = input$codes_to_remove + glob$pool, + glob$active_project, + user_id = glob$user$user_id, + doc_id = input$doc_selector, + segment_id = input$codes_to_remove ) removeModal() - loc$text <- load_doc_to_display( - glob$pool, - glob$active_project, - user = glob$user, - input$doc_selector, - loc$code_df$active_codebook, - ns = NS(id) - ) + # Refresh text + loc$text_observer <- loc$text_observer + 1 + # Notify analysis screen glob$segments_observer <- glob$segments_observer + 1 } }) - # # Helper (to be commented out in prod): position counter --------------- - + # Helper: position counter --------------- output$captured_range <- renderText({ - input$tag_position + req(isTruthy(input$tag_position)) + splitted_range <- strsplit(input$tag_position, split = "-") + if (splitted_range[[1]][1] == splitted_range[[1]][2]) { + reported_range <- unique(splitted_range[[1]]) + } else { + reported_range <- input$tag_position + } + paste(reported_range) }) - # returns glob$segments_observer + # returns glob$segments_observer and glob$codebook }) } diff --git a/R/mod_document_code_utils_document_code.R b/R/mod_document_code_utils_document_code.R index d0176043..b13ed094 100644 --- a/R/mod_document_code_utils_document_code.R +++ b/R/mod_document_code_utils_document_code.R @@ -75,7 +75,8 @@ load_segments_db <- function(pool, active_project, user, doc_id) { segments <- dplyr::tbl(pool, "segments") %>% dplyr::filter(.data$project_id == as.integer(active_project), .data$doc_id == as.integer(.env$doc_id)) %>% - dplyr::select(code_id, + dplyr::select(segment_id, + code_id, segment_start, segment_end, user_id) %>% @@ -87,7 +88,7 @@ load_segments_db <- function(pool, active_project, user, doc_id) { } return(segments %>% - dplyr::select(code_id, segment_start, segment_end)) + dplyr::select(segment_id, code_id, segment_start, segment_end)) } else {""} } @@ -230,6 +231,7 @@ calculate_code_overlap <- function(raw_segments) { names(vals) <- prevals$name res <- dplyr::tibble( + segment_id = NULL, code_id = NULL, segment_start = NULL, segment_end = NULL @@ -268,8 +270,7 @@ calculate_code_overlap <- function(raw_segments) { dplyr::filter( code_id != "", !is.na(code_id) - ) %>% - dplyr::mutate(segment_id = 0:(dplyr::n() - 1)) + ) } else { res } @@ -282,8 +283,8 @@ load_doc_to_display <- function(pool, user, doc_selector, codebook, + highlight, ns){ - position_type <- position_start <- tag_start <- tag_end <- NULL raw_text <- load_doc_db(pool, active_project, doc_selector) @@ -327,16 +328,20 @@ load_doc_to_display <- function(pool, dplyr::mutate(tag_end = "", tag_start = paste0('')) %>% + '" onclick="Shiny.setInputValue(\'', ns("clicked_title"), '\', this.title, {priority: \'event\'});">')) %>% dplyr::bind_rows( # start doc tibble::tibble(position_start = 0, position_type = "segment_start", - tag_start = "

"), + tag_start = "

"), # content ., # end doc @@ -377,7 +382,7 @@ load_doc_to_display <- function(pool, }else{ df_non_coded <- paste0( - "

", + "

", htmltools::htmlEscape(raw_text) %>% stringr::str_replace_all("[\\n\\r]", @@ -513,3 +518,15 @@ blend_colors <- function(string_id, code_names) { paste0("rgb(", color_mean_string, ")") } + +# highlight/underline ---- + +highlight_style <- function(choice) { + + switch(choice, + underline = '" class="segment" style="padding:0; text-decoration: underline; text-decoration-color:', + background = '" class="segment" style="padding:0; background-color:', + ) + + +} \ No newline at end of file diff --git a/R/mod_memo_utils_memo.R b/R/mod_memo_utils_memo.R index 1549fb31..b54691f3 100644 --- a/R/mod_memo_utils_memo.R +++ b/R/mod_memo_utils_memo.R @@ -133,11 +133,10 @@ export_memos <- function(pool, project) { } + # dropdown2 function ---- dropdownBlock2 <- function (..., id, icon = NULL, title = NULL, badgeStatus = "danger") { - if (!is.null(badgeStatus)) - shinydashboard:::validateStatus(badgeStatus) items <- c(list(...)) dropdownClass <- paste0("dropdown") numItems <- length(items) diff --git a/R/mod_project.R b/R/mod_project.R index b35bafc0..7fab80c1 100644 --- a/R/mod_project.R +++ b/R/mod_project.R @@ -19,16 +19,23 @@ mod_project_ui <- function(id) { tagList( fluidRow( class = "module_tools", - div(mod_rql_button_ui(ns("project_edit_tool"), - label = "Edit project", - icon = "pencil", - inputId = ns("project_edit_menu") - )) %>% tagAppendAttributes(style = "padding-right: 25px;"), - mod_rql_button_ui(ns("project_delete_tool"), + div( + mod_rql_button_ui(ns("project_edit_tool"), + label = "Edit project", + icon = "pencil", + inputId = ns("project_edit_menu") + ) + ) %>% tagAppendAttributes(style = "padding-right: 25px;"), + div(mod_rql_button_ui(ns("project_delete_tool"), label = "Delete project", icon = "trash", inputId = ns("project_delete_menu") - ) + )) %>% tagAppendAttributes(style = "padding-right: 25px;") #, + #mod_rql_button_ui(ns("project_import_tool"), + # label = "Import project", + # icon = "file-import", + # inputId = ns("project_import_menu") + #) ), fluidRow( class = "module_content", @@ -70,15 +77,15 @@ mod_project_server <- function(id, glob) { ns <- session$ns loc <- reactiveValues() loc$edit_observer <- 0 - output$project_name <- renderText({ - loc$project_name - }) - output$project_description <- renderText({ - loc$project_description - }) - output$project_date <- renderText({ - loc$project_date - }) + output$project_name <- renderText({ + loc$project_name + }) + output$project_description <- renderText({ + loc$project_description + }) + output$project_date <- renderText({ + loc$project_date + }) observeEvent(glob$active_project, { loc$project_df <- dplyr::tbl(glob$pool, "projects") %>% dplyr::filter(project_id == local(as.integer(glob$active_project))) @@ -104,33 +111,35 @@ mod_project_server <- function(id, glob) { ), glob, permission = "project_owner" - ) + ) observeEvent(input$project_edit_save, { active_project <- as.integer(local(glob$active_project)) if (input$project_edit_name != loc$project_name) { res <- DBI::dbExecute( - glob$pool, - glue::glue_sql("UPDATE projects + glob$pool, + glue::glue_sql("UPDATE projects SET project_name = {input$project_edit_name} WHERE project_id = {active_project}", .con = glob$pool) ) - updateTextInput(session = session, - "project_edit_name", - value = input$project_edit_name + updateTextInput( + session = session, + "project_edit_name", + value = input$project_edit_name ) } if (input$project_edit_description != loc$project_description) { res <- DBI::dbExecute( - glob$pool, - glue::glue_sql("UPDATE projects + glob$pool, + glue::glue_sql("UPDATE projects SET project_description = {input$project_edit_description} WHERE project_id = {active_project}", .con = glob$pool) ) - updateTextInput(session = session, - "project_edit_description", - value = input$project_edit_description + updateTextInput( + session = session, + "project_edit_description", + value = input$project_edit_description ) } @@ -138,7 +147,6 @@ mod_project_server <- function(id, glob) { dplyr::filter(project_id == local(as.integer(glob$active_project))) shinyjs::removeClass(paste0("sw-content-", ns("project_edit_menu")), "sw-show", asis = TRUE) showNotification("Changes to project were saved.") - }) # Project delete UI ---- diff --git a/R/rql_logo.R b/R/rql_logo.R new file mode 100644 index 00000000..949f3e66 --- /dev/null +++ b/R/rql_logo.R @@ -0,0 +1,3 @@ +rql_logo <- function(){ + "" +} \ No newline at end of file diff --git a/R/utils.R b/R/utils.R index fdd63e80..776dae18 100644 --- a/R/utils.R +++ b/R/utils.R @@ -50,6 +50,11 @@ utils::globalVariables(c("project_name", "credentials" )) +# dummy function for satisfying checks (getting rid of Note on not used imports) +dummy <- function(){ + dbplyr::sql + RPostgreSQL::dbConnect +} set_dashboard_body <- function() { @@ -334,6 +339,15 @@ warn_user <- function(warning) { warning)) } +# send message to interactive or Shiny session +rql_message <- function(msg) { + if (shiny::isRunning()){ + showNotification(msg) + } else { + message(msg) + } +} + # check permission to modify permissions check_modify_permission <- function(permission, msg) { @@ -394,6 +408,38 @@ rql_button_UI <- function(inputId, label, class = NULL) { tagAppendAttributes(style = "text-align: left;") } -rql_logo <- function(){ - "" + + + +db_helper_column <- function(pool, table, column, action){ + + check_colnames <- colnames(dplyr::tbl(pool, table)) + query <- switch(action, + "add" = glue::glue_sql(" + ALTER TABLE {`table`} + ADD COLUMN {`column`} INTEGER; + ", .con = pool), + "drop" = glue::glue_sql(" + ALTER TABLE {`table`} + DROP COLUMN {`column`} + ", .con = pool) + ) + if (!column %in% check_colnames && action == "add") { + res <- DBI::dbExecute(pool, query) + } else if (column %in% check_colnames && action == "drop"){ + res <- DBI::dbExecute(pool, query) + } else { + NULL + } } + +db_update_value <- function(pool, table, col_val, by_col_val){ + # col_val can be a list - list(c(col=1), c(col=2)) + query <- purrr::map(col_val, .f = function(x){ + glue::glue_sql("UPDATE {table} + SET {names(x)} = {x} + WHERE {names(by_col_val)} = {by_col_val}", .con = pool)}) + + res <- purrr::map(query, ~tryCatch({DBI::dbExecute(pool, .x)})) + +} \ No newline at end of file diff --git a/dev/run_dev.R b/dev/run_dev.R index 0608b98c..ea8f1683 100644 --- a/dev/run_dev.R +++ b/dev/run_dev.R @@ -8,32 +8,31 @@ golem::detach_all_attached() # Document and reload your package golem::document_and_reload() -# Run the application -# (run_app( -# mode = "server", -# dbname = "requal", -# dbhost = "localhost", -# dbusername = "requal_admin", -# dbpassword = "test", -# credentials_path = "requal_users.sqlite", -# credentials_pass = "test", -# options = list("launch.browser") -# )) - - (run_app( - mode = "local", + mode = "server", + dbname = "requal", + dbhost = "localhost", + dbusername = "requal_admin", + dbpassword = "test", + credentials_path = "requal_users.sqlite", + credentials_pass = "test", options = list("launch.browser") -)) + )) -(run_app( - mode = "local_test", - dbname = "tests/test.requal", - # dbhost = "localhost", - # dbusername = "requal_admin", - # dbpassword = "test", - # credentials_path = "requal_users.sqlite", - # credentials_pass = "test", - options = list("launch.browser") -)) +# (run_app( +# mode = "local", +# options = list("launch.browser") +#)) + + +# (run_app( +# mode = "local_test", +# dbname = "tests/test.requal", +# # dbhost = "localhost", +# # dbusername = "requal_admin", +# # dbpassword = "test", +# # credentials_path = "requal_users.sqlite", +# # credentials_pass = "test", +# options = list("launch.browser") +# )) diff --git a/inst/app/www/custom.css b/inst/app/www/custom.css index 4fb3fa48..8eeb9306 100644 --- a/inst/app/www/custom.css +++ b/inst/app/www/custom.css @@ -33,8 +33,14 @@ .code-button:hover { overflow: visible; + white-space: normal; + text-overflow: clip; +} +.code-button:popover { + overflow: initial; + white-space: initial; + text-overflow: initial; } - .quote { font-size: medium; line-height: 1; @@ -60,12 +66,14 @@ p.docpar { } .br { -height: 0px; -width: 0px; + height: 0px; + width: 0px; } b.segment { -font-weight: normal; + font-weight: normal; + text-decoration-thickness: 5px !important; + text-underline-offset: 3px !important; } @@ -150,3 +158,60 @@ width: 100% !important; .logo { background-color: rgb(63, 79, 144) !important; } +.split { + display: flex; + flex-direction: row; +} + +.split > #split-1 { + padding-top: 5px !important; + padding-left: 10px !important; + padding-right: 10px !important; + background-color: #ffffff !important; +} +.gutter { + background-color: #eee; + background-repeat: no-repeat; + background-position: 50%; +} + +.gutter.gutter-horizontal { + background-image: url(''); + cursor: col-resize; +} + +#rank-list-codes_menu { +background-color: transparent; +border: none; +padding: 0px; +margin-left: -0.3em; +margin-right: -0.3em; +display: flex; +} + + +.docpar::selection { + background: #ffffcc; + color: black; + text-decoration-line: underline; + text-decoration-style: dotted; + text-decoration-color: red; +} + +.docpar b::selection { + background: rgb(255, 255, 204, 0.75); + color: black; + text-decoration-line: underline; + text-decoration-style: dotted; + text-decoration-color: red; +} + +.quickcode { + height: calc(1.5em + .75rem + 2px); +} + + /* codes menu column layout */ + .two_columns { + display: grid; + grid-template-columns: 1fr 1fr; /* Creates two columns of equal width */ + } diff --git a/inst/app/www/document_code_js.js b/inst/app/www/document_code_js.js index 1e87e921..06a8db83 100644 --- a/inst/app/www/document_code_js.js +++ b/inst/app/www/document_code_js.js @@ -69,25 +69,57 @@ document.addEventListener('mouseup', function () { } var tag_position_value = startOffset.toString() + '-' + endOffset.toString(); - + console.log("tag_position" + tag_position_value) Shiny.setInputValue('document_code_ui_1-tag_position', tag_position_value); } }, false); }) -// Auxiliary function to add highlight in the browser + +Shiny.addCustomMessageHandler('getIframeContent', function(message) { + var iframe = document.getElementsByTagName('iframe')[0]; + var res = iframe.contentDocument.getElementById('quickCodeInput'); + var quickodeValue = res.dataset.quickode; + Shiny.setInputValue('document_code_ui_1-quickcode', quickodeValue); +}); + +Shiny.addCustomMessageHandler('refreshIframe', function(message) { + var iframe = document.getElementsByTagName('iframe')[0]; + iframe.src = iframe.src; +}); + +function findScrollElement(message) { + let targetStart = parseInt(message, 10); + console.log("Target Start:", targetStart); + let segments = document.querySelectorAll('article .segment'); + let segmentStartValues = Array.from(segments).map(el => parseInt(el.dataset.segment_start, 10)); + let index = segmentStartValues.findIndex(value => value => targetStart); + return segments[index]; // This could be undefined if no matching segment is found +} $( document ).ready(function() { - Shiny.addCustomMessageHandler('highlight', function(arg_color) { - - var selection = window.getSelection().getRangeAt(0); - if(window.getSelection().baseNode.parentNode.id != "document_code_ui_1-focal_text") return; - var selectedText = selection.extractContents(); - var mark = document.createElement("mark"); - mark.style.background = arg_color; - mark.appendChild(selectedText); - selection.insertNode(mark); - }) + +Shiny.addCustomMessageHandler('scrollToSegment', function(message) { + let el = findScrollElement(message); + if (el) { // Check if the element exists before trying to scroll into view + console.log("Scrolling to element:", el); + scrollToElementWithinContainer(el); + } else { + console.log("No element found to scroll to for message:", message); + } +}); }); +function scrollToElementWithinContainer(targetSelected) { + let container = document.querySelector('#document_code_ui_1-focal_text'); + let target = targetSelected; + + if (container && target) { + let targetPosition = target.getBoundingClientRect().top; + let containerPosition = container.getBoundingClientRect().top; + let scrollPosition = targetPosition - containerPosition + container.scrollTop; + + container.scrollTo({ top: scrollPosition, behavior: 'smooth' }); + } +} diff --git a/inst/app/www/highlight_style.js b/inst/app/www/highlight_style.js new file mode 100644 index 00000000..7d7a7626 --- /dev/null +++ b/inst/app/www/highlight_style.js @@ -0,0 +1,20 @@ +$( document ).ready(function(event) { + // Listen for messages of type 'toggleStyle' + Shiny.addCustomMessageHandler('toggleStyle', function(mode) { + // Loop over all elements with the class 'segment' + $('.segment').each(function() { + var higlight_tag = $(this); + var original_background_color = higlight_tag.data('color'); // Get the original color from the data-color attribute + // Determine the new mode and apply styles accordingly + if (mode === 'underline') { + higlight_tag.css('background-color', 'transparent'); // Clear background color + higlight_tag.css('text-decoration', 'underline'); // Apply underline + higlight_tag.css('text-decoration-color', original_background_color); // Use the original color for underline + } else if (mode === 'background') { + higlight_tag.css('background-color', original_background_color); // Apply original color as background color + higlight_tag.css('text-decoration', ''); // Remove underline + higlight_tag.css('text-decoration-color', ''); // Remove underline color + } + }); + }); +}); diff --git a/inst/app/www/highlight_tool.js b/inst/app/www/highlight_tool.js new file mode 100644 index 00000000..47e0c909 --- /dev/null +++ b/inst/app/www/highlight_tool.js @@ -0,0 +1,165 @@ + +Shiny.addCustomMessageHandler('wrapTextWithBold', function(message) { + const startOffset = message.startOffset; + const endOffset = message.endOffset; + const newId = message.newId; + const color = message.color; + const title = message.title; + wrapTextWithBold(startOffset, endOffset, newId, color, title); + }); + + + function wrapTextWithBold(startOffset, endOffset, newId, color, title) { + const container = document.querySelector('#article'); + if (!container) { + console.error("Container not found"); + return; + } + + let globalOffset = 0; + + Array.from(container.childNodes).forEach((p) => { + if (p.nodeType === Node.ELEMENT_NODE && p.tagName.toLowerCase() === 'p') { + let plainText = ''; + let bTagPositions = []; + Array.from(p.childNodes).forEach(child => { + if (child.nodeType === Node.ELEMENT_NODE && child.tagName.toLowerCase() === 'b') { + bTagPositions.push({pos: plainText.length, type: 'start', id: child.id, color: child.getAttribute('data-color'), title: child.title}); + plainText += child.textContent; + bTagPositions.push({pos: plainText.length, type: 'end', id: child.id}); + } else if (child.nodeType === Node.TEXT_NODE) { + plainText += child.textContent; + } + }); + + if (globalOffset <= endOffset && globalOffset + plainText.length >= startOffset) { + const paragraphStart = Math.max(startOffset - globalOffset, 0); + const paragraphEnd = Math.min(endOffset - globalOffset, plainText.length); + bTagPositions.push({pos: paragraphStart, type: 'start', id: newId, color: color, title: title}); + bTagPositions.push({pos: paragraphEnd, type: 'end', id: newId}); + } + + bTagPositions.sort((a, b) => a.pos - b.pos || (a.type === 'end' ? -1 : 1)); + + let finalBTagPositions = []; + let openTags = []; + let openColors = []; + let openTitles = []; + for (let i = 0; i < bTagPositions.length - 1; i++) { + if (bTagPositions[i].type === 'start') { + openTags.push(bTagPositions[i].id); + openColors.push(bTagPositions[i].color); + openTitles.push(bTagPositions[i].title); + } else { + const index = openTags.indexOf(bTagPositions[i].id); + openTags.splice(index, 1); + openColors.splice(index, 1); + openTitles.splice(index, 1); + } + if (bTagPositions[i].pos !== bTagPositions[i+1].pos && openTags.length > 0) { + let avgColor = "rgb(0,0,0)"; + if (openColors.length > 0) { + let r = 0, g = 0, b = 0; + openColors.forEach(color => { + let match = color.match(/\d+/g); + r += parseInt(match[0]); + g += parseInt(match[1]); + b += parseInt(match[2]); + }); + r = Math.round(r / openColors.length); + g = Math.round(g / openColors.length); + b = Math.round(b / openColors.length); + avgColor = `rgb(${r},${g},${b})`; + } + finalBTagPositions.push({start: bTagPositions[i].pos, end: bTagPositions[i+1].pos, id: openTags.join('+'), color: avgColor, title: openTitles.join(' | ')}); + } + } + + p.innerHTML = ''; + let currentOffset = 0; + finalBTagPositions.forEach((position) => { + if (position.start !== position.end) { + const before = document.createTextNode(plainText.slice(currentOffset, position.start)); + const middle = document.createElement('b'); + middle.id = position.id; + middle.title = position.title; + middle.setAttribute('data-color', position.color); + middle.classList.add('segment'); + middle.textContent = plainText.slice(position.start, position.end); + middle.setAttribute('onclick', "Shiny.setInputValue('document_code_ui_1-clicked_title', this.title, {priority: 'event'});"); + p.appendChild(before); + p.appendChild(middle); + currentOffset = position.end; + } + }); + const after = document.createTextNode(plainText.slice(currentOffset)); + p.appendChild(after); + + globalOffset += plainText.length; + } + }); + } + // // Auxiliary function to add highlight in the browser + // $( document ).ready(function() { + // Shiny.addCustomMessageHandler('highlight', function(message) { + // var startOff = message.startOff; + // var endOff = message.endOff; + // console.log(startOff) + // console.log(endOff) + // wrapSelection(startOff, endOff); + // }) + // }); + + // function wrapSelection(startOffset, endOffset) { + // let currentOffset = 0; + + // function traverse(node) { + // if (node.nodeType === Node.TEXT_NODE) { + // const textLength = node.textContent.length; + + // // If the startOffset is within this text node + // if (startOffset >= currentOffset && startOffset < currentOffset + textLength) { + // const start = startOffset - currentOffset; + // const before = node.textContent.slice(0, start); + // const after = node.textContent.slice(start); + // const newNode = document.createElement('b'); + // newNode.textContent = after; + // node.textContent = before; + // node.parentNode.insertBefore(newNode, node.nextSibling); + // } + + // // If the endOffset is within this text node + // if (endOffset > currentOffset && endOffset <= currentOffset + textLength) { + // const end = endOffset - currentOffset; + // const boldNode = node.nextSibling; + // const before = boldNode.textContent.slice(0, end); + // const after = boldNode.textContent.slice(end); + // const newNode = document.createTextNode(after); + // boldNode.textContent = before; + // boldNode.parentNode.insertBefore(newNode, boldNode.nextSibling); + // } + + // currentOffset += textLength; + // } else { + // for (let child of node.childNodes) { + // traverse(child); + // } + // } + // } + + // traverse(document.querySelector('article')); + // } + + //

+ //

Text of paragraph 1

+ //

Text of paragraph 2

+ //

Text of paragraph 3

+ //

Text of paragraph 4

+ //
+ + //
+ //

Text of paragraph 1

+ //

Text of paragraph 2

+ //

Text of paragraph 3

+ //

Text of paragraph 4

+ //
\ No newline at end of file diff --git a/inst/app/www/quickcode.html b/inst/app/www/quickcode.html new file mode 100644 index 00000000..cbc20b6f --- /dev/null +++ b/inst/app/www/quickcode.html @@ -0,0 +1,34 @@ + + + + + + + \ No newline at end of file diff --git a/inst/app/www/split.min.js b/inst/app/www/split.min.js new file mode 100644 index 00000000..cc0a2a43 --- /dev/null +++ b/inst/app/www/split.min.js @@ -0,0 +1,267 @@ +/*! Split.js - v1.6.0 */ +!function(e,t){"object"==typeof exports&&"undefined"!=typeof module?module.exports=t():"function"==typeof define&&define.amd?define(t):(e=e||self).Split=t()}(this,(function(){ + "use strict"; + var e="undefined"!=typeof window?window:null, + t=null===e, + n=t?void 0:e.document, + i=function(){return!1}, + r=t?"calc":["","-webkit-","-moz-","-o-"].filter((function(e){ + var t=n.createElement("div"); + return t.style.cssText="width:"+e+"calc(9px)", + !!t.style.length + })).shift()+"calc", + s=function(e){return"string"==typeof e||e instanceof String}, + o=function(e){if(s(e)){ + var t=n.querySelector(e); + if(!t)throw new Error("Selector "+e+" did not match a DOM element"); + return t + } + return e + }, + a=function(e,t,n){var i=e[t]; + return void 0!==i?i:n + }, + u=function(e,t,n,i){ + if(t){ + if("end"===i)return 0; + if("center"===i)return e/2 + }else if(n){ + if("start"===i)return 0; + if("center"===i)return e/2 + } + return e + }, + l=function(e,t){var i=n.createElement("div"); + return i.className="gutter gutter-"+t,i + }, + c=function(e,t,n){var i={}; + return s(t)?i[e]=t:i[e]=r+"("+t+"% - "+n+"px)",i + }, + h=function(e,t){var n; + return(n={})[e]=t+"px",n + }; + return function(r,s){ + if(void 0===s&&(s={}),t)return{}; + var d,f,v,m,g,p,y=r; + Array.from&&(y=Array.from(y)); + var z=o(y[0]).parentNode, + b=getComputedStyle?getComputedStyle(z):null, + E=b?b.flexDirection:null, + S=a(s,"sizes")||y.map((function(){return 100/y.length})), + L=a(s,"minSize",100), + _=Array.isArray(L)?L:y.map((function(){return L})), + w=a(s,"expandToMin",!1), + k=a(s,"gutterSize",10), + x=a(s,"gutterAlign","center"), + C=a(s,"snapOffset",30), + M=a(s,"dragInterval",1), + U=a(s,"direction","horizontal"), + O=a(s,"cursor","horizontal"===U?"col-resize":"row-resize"), + D=a(s,"gutter",l), + A=a(s,"elementStyle",c), + B=a(s,"gutterStyle",h); + function j(e,t,n,i){ + var r=A(d,t,n,i); + Object.keys(r).forEach((function(t){ + e.style[t]=r[t] + })) + } + function F(){ + return p.map((function(e){ + return e.size + })) + } + function R(e){ + return"touches"in e?e.touches[0][f]:e[f] + } + function T(e){ + var t=p[this.a], + n=p[this.b], + i=t.size+n.size; + t.size=e/this.size*i, + n.size=i-e/this.size*i, + j(t.element,t.size,this._b,t.i), + j(n.element,n.size,this._c,n.i) + } + function N(e){ + var t,n=p[this.a], + r=p[this.b]; + this.dragging&&(t=R(e)-this.start+(this._b-this.dragOffset), + M>1&&(t=Math.round(t/M)*M), + t<=n.minSize+C+this._b?t=n.minSize+this._b:t>=this.size-(r.minSize+C+this._c)&&(t=this.size-(r.minSize+this._c)), + T.call(this,t), + a(s,"onDrag",i)()) + } + function q(){ + var e=p[this.a].element, + t=p[this.b].element, + n=e.getBoundingClientRect(), + i=t.getBoundingClientRect(); + this.size=n[d]+i[d]+this._b+this._c, + this.start=n[v], + this.end=n[m] + } + function H(e){ + var t=function(e){ + if(!getComputedStyle)return null; + var t=getComputedStyle(e); + if(!t)return null; + var n=e[g]; + return 0===n?null:n-="horizontal"===U?parseFloat(t.paddingLeft)+parseFloat(t.paddingRight):parseFloat(t.paddingTop)+parseFloat(t.paddingBottom) + }(z); + if(null===t)return e; + if(_.reduce((function(e,t){return e+t}),0)>t)return e; + var n=0,i=[],r=e.map((function(r,s){ + var o=t*r/100, + a=u(k,0===s,s===e.length-1,x), + l=_[s]+a; + return o0&&i[r]-n>0){ + var o=Math.min(n,i[r]-n); + n-=o,s=e-o + } + return s/t*100 + })) + } + function I(){ + var t=p[this.a].element, + r=p[this.b].element; + this.dragging&&a(s,"onDragEnd",i)(F()), + this.dragging=!1, + e.removeEventListener("mouseup",this.stop), + e.removeEventListener("touchend",this.stop), + e.removeEventListener("touchcancel",this.stop), + e.removeEventListener("mousemove",this.move), + e.removeEventListener("touchmove",this.move), + this.stop=null, + this.move=null, + t.removeEventListener("selectstart",i), + t.removeEventListener("dragstart",i), + r.removeEventListener("selectstart",i), + r.removeEventListener("dragstart",i), + t.style.userSelect="", + t.style.webkitUserSelect="", + t.style.MozUserSelect="", + t.style.pointerEvents="", + r.style.userSelect="", + r.style.webkitUserSelect="", + r.style.MozUserSelect="", + r.style.pointerEvents="", + this.gutter.style.cursor="", + this.parent.style.cursor="", + n.body.style.cursor="" + } + function W(t){ + if(!("button"in t)||0===t.button){ + var r=p[this.a].element, + o=p[this.b].element; + this.dragging||a(s,"onDragStart",i)(F()), + t.preventDefault(), + this.dragging=!0, + this.move=N.bind(this), + this.stop=I.bind(this), + e.addEventListener("mouseup",this.stop), + e.addEventListener("touchend",this.stop), + e.addEventListener("touchcancel",this.stop), + e.addEventListener("mousemove",this.move), + e.addEventListener("touchmove",this.move), + r.addEventListener("selectstart",i), + r.addEventListener("dragstart",i), + o.addEventListener("selectstart",i), + o.addEventListener("dragstart",i), + r.style.userSelect="none", + r.style.webkitUserSelect="none", + r.style.MozUserSelect="none", + r.style.pointerEvents="none", + o.style.userSelect="none", + o.style.webkitUserSelect="none", + o.style.MozUserSelect="none", + o.style.pointerEvents="none", + this.gutter.style.cursor=O, + this.parent.style.cursor=O, + n.body.style.cursor=O, + q.call(this), + this.dragOffset=R(t)-this.end + } + } + "horizontal"===U?(d="width",f="clientX",v="left",m="right",g="clientWidth"):"vertical"===U&&(d="height",f="clientY",v="top",m="bottom",g="clientHeight"), + S=H(S); + var X=[]; + function Y(e){ + var t=e.i===X.length,n=t?X[e.i-1]:X[e.i]; + q.call(n); + var i=t?n.size-e.minSize-n._c:e.minSize+n._b; + T.call(n,i) + } + return(p=y.map((function(e,t){ + var n,i={element:o(e),size:S[t],minSize:_[t],i:t}; + if(t>0&&((n={a:t-1,b:t,dragging:!1,direction:U,parent:z})._b=u(k,t-1==0,!1,x), + n._c=u(k,!1,t===y.length-1,x), + "row-reverse"===E||"column-reverse"===E)){ + var r=n.a; + n.a=n.b, + n.b=r + } + if(t>0){ + var s=D(t,U,i.element); + !function(e,t,n){ + var i=B(d,t,n); + Object.keys(i).forEach((function(t){ + e.style[t]=i[t] + })) + }(s,k,t), + n._a=W.bind(n), + s.addEventListener("mousedown",n._a), + s.addEventListener("touchstart",n._a), + z.insertBefore(s,i.element), + n.gutter=s + } + return j(i.element,i.size,u(k,0===t,t===y.length-1,x),t), + t>0&&X.push(n), + i + }))).forEach((function(e){ + var t=e.element.getBoundingClientRect()[d]; + t0){ + var i=X[n-1], + r=p[i.a], + s=p[i.b]; + r.size=t[n-1], + s.size=e, + j(r.element,r.size,i._b,i.i), + j(s.element,s.size,i._c,s.i) + } + })) + }, + getSizes:F, + collapse:function(e){ + Y(p[e]) + }, + destroy:function(e,t){ + X.forEach((function(n){ + if(!0!==t?n.parent.removeChild(n.gutter):(n.gutter.removeEventListener("mousedown",n._a), + n.gutter.removeEventListener("touchstart",n._a)), + !0!==e){ + var i=A(d,n.a.size,n._b); + Object.keys(i).forEach((function(e){ + p[n.a].element.style[e]="", + p[n.b].element.style[e]="" + })) + } + })) + }, + parent:z, + pairs:X + } + } +})); + + diff --git a/inst/refi.xsd b/inst/refi.xsd new file mode 100644 index 00000000..a2552ff6 --- /dev/null +++ b/inst/refi.xsd @@ -0,0 +1,501 @@ + + + + + + + + + + This element MUST be conveyed as the root element in any instance document based on this Schema expression + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/inst/test_app/tests/testthat/_snaps/app-create-codes/createcode-001_.png b/inst/test_app/tests/testthat/_snaps/app-create-codes/createcode-001_.png index d7adc8b2..a7775a45 100644 Binary files a/inst/test_app/tests/testthat/_snaps/app-create-codes/createcode-001_.png and b/inst/test_app/tests/testthat/_snaps/app-create-codes/createcode-001_.png differ diff --git a/inst/test_app/tests/testthat/_snaps/app-create-codes/createcode-002_.png b/inst/test_app/tests/testthat/_snaps/app-create-codes/createcode-002_.png index 230543b0..70f732ed 100644 Binary files a/inst/test_app/tests/testthat/_snaps/app-create-codes/createcode-002_.png and b/inst/test_app/tests/testthat/_snaps/app-create-codes/createcode-002_.png differ diff --git a/inst/test_app/tests/testthat/_snaps/app-delete-code/requaltest-001_.png b/inst/test_app/tests/testthat/_snaps/app-delete-code/requaltest-001_.png index daf82135..66236e12 100644 Binary files a/inst/test_app/tests/testthat/_snaps/app-delete-code/requaltest-001_.png and b/inst/test_app/tests/testthat/_snaps/app-delete-code/requaltest-001_.png differ diff --git a/inst/test_app/tests/testthat/_snaps/app-delete-code/requaltest-002_.png b/inst/test_app/tests/testthat/_snaps/app-delete-code/requaltest-002_.png index aa4b4c38..bf37d81c 100644 Binary files a/inst/test_app/tests/testthat/_snaps/app-delete-code/requaltest-002_.png and b/inst/test_app/tests/testthat/_snaps/app-delete-code/requaltest-002_.png differ diff --git a/inst/test_app/tests/testthat/_snaps/app-edit-code/edit_code-001.json b/inst/test_app/tests/testthat/_snaps/app-edit-code/edit_code-001.json new file mode 100644 index 00000000..06cb876d --- /dev/null +++ b/inst/test_app/tests/testthat/_snaps/app-edit-code/edit_code-001.json @@ -0,0 +1,25 @@ +{ + "output": { + "codebook_ui_1-codes_ui": { + "html": "
\n
\n
\n

Code1<\/h3>\n
\n code<\/span>\n