Compare commits
	
		
			56 Commits
		
	
	
		
			ed4f1d6f2c
			...
			v0.2025.9
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 3ae4b7086a | |||
| 446b1f8600 | |||
| 00fd208a2c | |||
| e574d03716 | |||
| c1f3116c9d | |||
| 3aec7e6c14 | |||
| 9f0020dec8 | |||
| 6e78ad9729 | |||
| 44d84a9b2a | |||
| ac7809415c | |||
| 675cecaa20 | |||
| 5d179cc088 | |||
| d905618590 | |||
| 3fd9bc0b18 | |||
| 39abee7f73 | |||
| b770619111 | |||
| 1c44857da4 | |||
| bca4440867 | |||
| 4855543961 | |||
| cb3d6a98b9 | |||
| ada67a13d3 | |||
| f4c928f26e | |||
| 91fd515d39 | |||
| be6e841d3d | |||
| af6afa6903 | |||
| 6ab5d2a28d | |||
| 4be033f288 | |||
| c550f92003 | |||
| ed836b3ee0 | |||
| ac7a43abf4 | |||
| 49f19fce91 | |||
| b2197eb8e9 | |||
| 5fbc2cae1c | |||
| 730abb49ce | |||
| edccab054a | |||
| e8210c6fdd | |||
| 55d69d7c13 | |||
| 50fb18d4ff | |||
| 77b1ea1fc8 | |||
| 61501a9b64 | |||
| c1507adac5 | |||
| 45fb9eda1c | |||
| 982a61f4bf | |||
| 18e5b41663 | |||
| 910c39cbd0 | |||
| 9952dfd49d | |||
| 00f75d5382 | |||
| d78828554b | |||
| b84b561109 | |||
| a618815500 | |||
| e1f3dc6ae4 | |||
| f378db6c6f | |||
| 7cec0f7d61 | |||
| f902d0374c | |||
| b5f0a0c4f7 | |||
| 00623cea09 | 
							
								
								
									
										357
									
								
								Doxyfile
									
									
									
									
									
								
							
							
						
						
									
										357
									
								
								Doxyfile
									
									
									
									
									
								
							| @@ -1,4 +1,4 @@ | ||||
| # Doxyfile 1.9.8 | ||||
| # Doxyfile 1.9.4 | ||||
|  | ||||
| # This file describes the settings to be used by the documentation system | ||||
| # doxygen (www.doxygen.org) for a project. | ||||
| @@ -19,8 +19,7 @@ | ||||
| # configuration file: | ||||
| # doxygen -x [configFile] | ||||
| # Use doxygen to compare the used configuration file with the template | ||||
| # configuration file without replacing the environment variables or CMake type | ||||
| # replacement variables: | ||||
| # configuration file without replacing the environment variables: | ||||
| # doxygen -x_noenv [configFile] | ||||
|  | ||||
| #--------------------------------------------------------------------------- | ||||
| @@ -86,7 +85,7 @@ CREATE_SUBDIRS         = NO | ||||
| # level increment doubles the number of directories, resulting in 4096 | ||||
| # directories at level 8 which is the default and also the maximum value. The | ||||
| # sub-directories are organized in 2 levels, the first level always has a fixed | ||||
| # number of 16 directories. | ||||
| # numer of 16 directories. | ||||
| # Minimum value: 0, maximum value: 8, default value: 8. | ||||
| # This tag requires that the tag CREATE_SUBDIRS is set to YES. | ||||
|  | ||||
| @@ -363,17 +362,6 @@ MARKDOWN_SUPPORT       = YES | ||||
|  | ||||
| TOC_INCLUDE_HEADINGS   = 5 | ||||
|  | ||||
| # The MARKDOWN_ID_STYLE tag can be used to specify the algorithm used to | ||||
| # generate identifiers for the Markdown headings. Note: Every identifier is | ||||
| # unique. | ||||
| # Possible values are: DOXYGEN use a fixed 'autotoc_md' string followed by a | ||||
| # sequence number starting at 0 and GITHUB use the lower case version of title | ||||
| # with any whitespace replaced by '-' and punctuation characters removed. | ||||
| # The default value is: DOXYGEN. | ||||
| # This tag requires that the tag MARKDOWN_SUPPORT is set to YES. | ||||
|  | ||||
| MARKDOWN_ID_STYLE      = DOXYGEN | ||||
|  | ||||
| # When enabled doxygen tries to link words that correspond to documented | ||||
| # classes, or namespaces to their corresponding documentation. Such a link can | ||||
| # be prevented in individual cases by putting a % sign in front of the word or | ||||
| @@ -498,14 +486,6 @@ LOOKUP_CACHE_SIZE      = 0 | ||||
|  | ||||
| NUM_PROC_THREADS       = 1 | ||||
|  | ||||
| # If the TIMESTAMP tag is set different from NO then each generated page will | ||||
| # contain the date or date and time when the page was generated. Setting this to | ||||
| # NO can help when comparing the output of multiple runs. | ||||
| # Possible values are: YES, NO, DATETIME and DATE. | ||||
| # The default value is: NO. | ||||
|  | ||||
| TIMESTAMP              = NO | ||||
|  | ||||
| #--------------------------------------------------------------------------- | ||||
| # Build related configuration options | ||||
| #--------------------------------------------------------------------------- | ||||
| @@ -587,8 +567,7 @@ HIDE_UNDOC_MEMBERS     = NO | ||||
| # If the HIDE_UNDOC_CLASSES tag is set to YES, doxygen will hide all | ||||
| # undocumented classes that are normally visible in the class hierarchy. If set | ||||
| # to NO, these classes will be included in the various overviews. This option | ||||
| # will also hide undocumented C++ concepts if enabled. This option has no effect | ||||
| # if EXTRACT_ALL is enabled. | ||||
| # has no effect if EXTRACT_ALL is enabled. | ||||
| # The default value is: NO. | ||||
|  | ||||
| HIDE_UNDOC_CLASSES     = NO | ||||
| @@ -626,8 +605,7 @@ INTERNAL_DOCS          = NO | ||||
| # Windows (including Cygwin) and MacOS, users should typically set this option | ||||
| # to NO, whereas on Linux or other Unix flavors it should typically be set to | ||||
| # YES. | ||||
| # Possible values are: SYSTEM, NO and YES. | ||||
| # The default value is: SYSTEM. | ||||
| # The default value is: system dependent. | ||||
|  | ||||
| CASE_SENSE_NAMES       = YES | ||||
|  | ||||
| @@ -879,26 +857,11 @@ WARN_IF_INCOMPLETE_DOC = YES | ||||
|  | ||||
| WARN_NO_PARAMDOC       = NO | ||||
|  | ||||
| # If WARN_IF_UNDOC_ENUM_VAL option is set to YES, doxygen will warn about | ||||
| # undocumented enumeration values. If set to NO, doxygen will accept | ||||
| # undocumented enumeration values. If EXTRACT_ALL is set to YES then this flag | ||||
| # will automatically be disabled. | ||||
| # The default value is: NO. | ||||
|  | ||||
| WARN_IF_UNDOC_ENUM_VAL = NO | ||||
|  | ||||
| # If the WARN_AS_ERROR tag is set to YES then doxygen will immediately stop when | ||||
| # a warning is encountered. If the WARN_AS_ERROR tag is set to FAIL_ON_WARNINGS | ||||
| # then doxygen will continue running as if WARN_AS_ERROR tag is set to NO, but | ||||
| # at the end of the doxygen process doxygen will return with a non-zero status. | ||||
| # If the WARN_AS_ERROR tag is set to FAIL_ON_WARNINGS_PRINT then doxygen behaves | ||||
| # like FAIL_ON_WARNINGS but in case no WARN_LOGFILE is defined doxygen will not | ||||
| # write the warning messages in between other messages but write them at the end | ||||
| # of a run, in case a WARN_LOGFILE is defined the warning messages will be | ||||
| # besides being in the defined file also be shown at the end of a run, unless | ||||
| # the WARN_LOGFILE is defined as - i.e. standard output (stdout) in that case | ||||
| # the behavior will remain as with the setting FAIL_ON_WARNINGS. | ||||
| # Possible values are: NO, YES, FAIL_ON_WARNINGS and FAIL_ON_WARNINGS_PRINT. | ||||
| # Possible values are: NO, YES and FAIL_ON_WARNINGS. | ||||
| # The default value is: NO. | ||||
|  | ||||
| WARN_AS_ERROR          = NO | ||||
| @@ -957,21 +920,10 @@ INPUT                  = README.md \ | ||||
| # libiconv (or the iconv built into libc) for the transcoding. See the libiconv | ||||
| # documentation (see: | ||||
| # https://www.gnu.org/software/libiconv/) for the list of possible encodings. | ||||
| # See also: INPUT_FILE_ENCODING | ||||
| # The default value is: UTF-8. | ||||
|  | ||||
| INPUT_ENCODING         = UTF-8 | ||||
|  | ||||
| # This tag can be used to specify the character encoding of the source files | ||||
| # that doxygen parses The INPUT_FILE_ENCODING tag can be used to specify | ||||
| # character encoding on a per file pattern basis. Doxygen will compare the file | ||||
| # name with each pattern and apply the encoding instead of the default | ||||
| # INPUT_ENCODING) if there is a match. The character encodings are a list of the | ||||
| # form: pattern=encoding (like *.php=ISO-8859-1). See cfg_input_encoding | ||||
| # "INPUT_ENCODING" for further information on supported encodings. | ||||
|  | ||||
| INPUT_FILE_ENCODING    = | ||||
|  | ||||
| # If the value of the INPUT tag contains directories, you can use the | ||||
| # FILE_PATTERNS tag to specify one or more wildcard patterns (like *.cpp and | ||||
| # *.h) to filter out the source-files in the directories. | ||||
| @@ -983,12 +935,12 @@ INPUT_FILE_ENCODING    = | ||||
| # Note the list of default checked file patterns might differ from the list of | ||||
| # default file extension mappings. | ||||
| # | ||||
| # If left blank the following patterns are tested:*.c, *.cc, *.cxx, *.cxxm, | ||||
| # *.cpp, *.cppm, *.c++, *.c++m, *.java, *.ii, *.ixx, *.ipp, *.i++, *.inl, *.idl, | ||||
| # *.ddl, *.odl, *.h, *.hh, *.hxx, *.hpp, *.h++, *.ixx, *.l, *.cs, *.d, *.php, | ||||
| # *.php4, *.php5, *.phtml, *.inc, *.m, *.markdown, *.md, *.mm, *.dox (to be | ||||
| # provided as doxygen C comment), *.py, *.pyw, *.f90, *.f95, *.f03, *.f08, | ||||
| # *.f18, *.f, *.for, *.vhd, *.vhdl, *.ucf, *.qsf and *.ice. | ||||
| # If left blank the following patterns are tested:*.c, *.cc, *.cxx, *.cpp, | ||||
| # *.c++, *.java, *.ii, *.ixx, *.ipp, *.i++, *.inl, *.idl, *.ddl, *.odl, *.h, | ||||
| # *.hh, *.hxx, *.hpp, *.h++, *.l, *.cs, *.d, *.php, *.php4, *.php5, *.phtml, | ||||
| # *.inc, *.m, *.markdown, *.md, *.mm, *.dox (to be provided as doxygen C | ||||
| # comment), *.py, *.pyw, *.f90, *.f95, *.f03, *.f08, *.f18, *.f, *.for, *.vhd, | ||||
| # *.vhdl, *.ucf, *.qsf and *.ice. | ||||
|  | ||||
| FILE_PATTERNS          = *.h \ | ||||
|                          *.js \ | ||||
| @@ -1030,6 +982,9 @@ EXCLUDE_PATTERNS       = | ||||
| # output. The symbol name can be a fully qualified name, a word, or if the | ||||
| # wildcard * is used, a substring. Examples: ANamespace, AClass, | ||||
| # ANamespace::AClass, ANamespace::*Test | ||||
| # | ||||
| # Note that the wildcards are matched against the file with absolute path, so to | ||||
| # exclude all test directories use the pattern */test/* | ||||
|  | ||||
| EXCLUDE_SYMBOLS        = | ||||
|  | ||||
| @@ -1074,11 +1029,6 @@ IMAGE_PATH             = docs/images/ | ||||
| # code is scanned, but not when the output code is generated. If lines are added | ||||
| # or removed, the anchors will not be placed correctly. | ||||
| # | ||||
| # Note that doxygen will use the data processed and written to standard output | ||||
| # for further processing, therefore nothing else, like debug statements or used | ||||
| # commands (so in case of a Windows batch file always use @echo OFF), should be | ||||
| # written to standard output. | ||||
| # | ||||
| # Note that for custom extensions or not directly supported extensions you also | ||||
| # need to set EXTENSION_MAPPING for the extension otherwise the files are not | ||||
| # properly processed by doxygen. | ||||
| @@ -1120,15 +1070,6 @@ FILTER_SOURCE_PATTERNS = | ||||
|  | ||||
| USE_MDFILE_AS_MAINPAGE = README.md | ||||
|  | ||||
| # The Fortran standard specifies that for fixed formatted Fortran code all | ||||
| # characters from position 72 are to be considered as comment. A common | ||||
| # extension is to allow longer lines before the automatic comment starts. The | ||||
| # setting FORTRAN_COMMENT_AFTER will also make it possible that longer lines can | ||||
| # be processed before the automatic comment starts. | ||||
| # Minimum value: 7, maximum value: 10000, default value: 72. | ||||
|  | ||||
| FORTRAN_COMMENT_AFTER  = 72 | ||||
|  | ||||
| #--------------------------------------------------------------------------- | ||||
| # Configuration options related to source browsing | ||||
| #--------------------------------------------------------------------------- | ||||
| @@ -1266,11 +1207,10 @@ CLANG_DATABASE_PATH    = | ||||
|  | ||||
| ALPHABETICAL_INDEX     = YES | ||||
|  | ||||
| # The IGNORE_PREFIX tag can be used to specify a prefix (or a list of prefixes) | ||||
| # that should be ignored while generating the index headers. The IGNORE_PREFIX | ||||
| # tag works for classes, function and member names. The entity will be placed in | ||||
| # the alphabetical list under the first letter of the entity name that remains | ||||
| # after removing the prefix. | ||||
| # In case all classes in a project start with a common prefix, all classes will | ||||
| # be put under the same header in the alphabetical index. The IGNORE_PREFIX tag | ||||
| # can be used to specify a prefix (or a list of prefixes) that should be ignored | ||||
| # while generating the index headers. | ||||
| # This tag requires that the tag ALPHABETICAL_INDEX is set to YES. | ||||
|  | ||||
| IGNORE_PREFIX          = | ||||
| @@ -1349,12 +1289,7 @@ HTML_STYLESHEET        = | ||||
| # Doxygen will copy the style sheet files to the output directory. | ||||
| # Note: The order of the extra style sheet files is of importance (e.g. the last | ||||
| # style sheet in the list overrules the setting of the previous ones in the | ||||
| # list). | ||||
| # Note: Since the styling of scrollbars can currently not be overruled in | ||||
| # Webkit/Chromium, the styling will be left out of the default doxygen.css if | ||||
| # one or more extra stylesheets have been specified. So if scrollbar | ||||
| # customization is desired it has to be added explicitly. For an example see the | ||||
| # documentation. | ||||
| # list). For an example see the documentation. | ||||
| # This tag requires that the tag GENERATE_HTML is set to YES. | ||||
|  | ||||
| HTML_EXTRA_STYLESHEET  = | ||||
| @@ -1369,19 +1304,6 @@ HTML_EXTRA_STYLESHEET  = | ||||
|  | ||||
| HTML_EXTRA_FILES       = | ||||
|  | ||||
| # The HTML_COLORSTYLE tag can be used to specify if the generated HTML output | ||||
| # should be rendered with a dark or light theme. | ||||
| # Possible values are: LIGHT always generate light mode output, DARK always | ||||
| # generate dark mode output, AUTO_LIGHT automatically set the mode according to | ||||
| # the user preference, use light mode if no preference is set (the default), | ||||
| # AUTO_DARK automatically set the mode according to the user preference, use | ||||
| # dark mode if no preference is set and TOGGLE allow to user to switch between | ||||
| # light and dark mode via a button. | ||||
| # The default value is: AUTO_LIGHT. | ||||
| # This tag requires that the tag GENERATE_HTML is set to YES. | ||||
|  | ||||
| HTML_COLORSTYLE        = AUTO_LIGHT | ||||
|  | ||||
| # The HTML_COLORSTYLE_HUE tag controls the color of the HTML output. Doxygen | ||||
| # will adjust the colors in the style sheet and background images according to | ||||
| # this color. Hue is specified as an angle on a color-wheel, see | ||||
| @@ -1412,6 +1334,15 @@ HTML_COLORSTYLE_SAT    = 100 | ||||
|  | ||||
| HTML_COLORSTYLE_GAMMA  = 80 | ||||
|  | ||||
| # If the HTML_TIMESTAMP tag is set to YES then the footer of each generated HTML | ||||
| # page will contain the date and time when the page was generated. Setting this | ||||
| # to YES can help to show when doxygen was last run and thus if the | ||||
| # documentation is up to date. | ||||
| # The default value is: NO. | ||||
| # This tag requires that the tag GENERATE_HTML is set to YES. | ||||
|  | ||||
| #HTML_TIMESTAMP         = NO | ||||
|  | ||||
| # If the HTML_DYNAMIC_MENUS tag is set to YES then the generated HTML | ||||
| # documentation will contain a main index with vertical navigation menus that | ||||
| # are dynamically created via JavaScript. If disabled, the navigation index will | ||||
| @@ -1431,13 +1362,6 @@ HTML_DYNAMIC_MENUS     = YES | ||||
|  | ||||
| HTML_DYNAMIC_SECTIONS  = NO | ||||
|  | ||||
| # If the HTML_CODE_FOLDING tag is set to YES then classes and functions can be | ||||
| # dynamically folded and expanded in the generated HTML source code. | ||||
| # The default value is: YES. | ||||
| # This tag requires that the tag GENERATE_HTML is set to YES. | ||||
|  | ||||
| HTML_CODE_FOLDING      = YES | ||||
|  | ||||
| # With HTML_INDEX_NUM_ENTRIES one can control the preferred number of entries | ||||
| # shown in the various tree structured indices initially; the user can expand | ||||
| # and collapse entries dynamically later on. Doxygen will expand the tree to | ||||
| @@ -1568,16 +1492,6 @@ BINARY_TOC             = NO | ||||
|  | ||||
| TOC_EXPAND             = NO | ||||
|  | ||||
| # The SITEMAP_URL tag is used to specify the full URL of the place where the | ||||
| # generated documentation will be placed on the server by the user during the | ||||
| # deployment of the documentation. The generated sitemap is called sitemap.xml | ||||
| # and placed on the directory specified by HTML_OUTPUT. In case no SITEMAP_URL | ||||
| # is specified no sitemap is generated. For information about the sitemap | ||||
| # protocol see https://www.sitemaps.org | ||||
| # This tag requires that the tag GENERATE_HTML is set to YES. | ||||
|  | ||||
| SITEMAP_URL            = | ||||
|  | ||||
| # If the GENERATE_QHP tag is set to YES and both QHP_NAMESPACE and | ||||
| # QHP_VIRTUAL_FOLDER are set, an additional index file will be generated that | ||||
| # can be used as input for Qt's qhelpgenerator to generate a Qt Compressed Help | ||||
| @@ -1753,6 +1667,17 @@ HTML_FORMULA_FORMAT    = png | ||||
|  | ||||
| FORMULA_FONTSIZE       = 10 | ||||
|  | ||||
| # Use the FORMULA_TRANSPARENT tag to determine whether or not the images | ||||
| # generated for formulas are transparent PNGs. Transparent PNGs are not | ||||
| # supported properly for IE 6.0, but are supported on all modern browsers. | ||||
| # | ||||
| # Note that when changing this option you need to delete any form_*.png files in | ||||
| # the HTML output directory before the changes have effect. | ||||
| # The default value is: YES. | ||||
| # This tag requires that the tag GENERATE_HTML is set to YES. | ||||
|  | ||||
| #FORMULA_TRANSPARENT    = YES | ||||
|  | ||||
| # The FORMULA_MACROFILE can contain LaTeX \newcommand and \renewcommand commands | ||||
| # to create new LaTeX commands to be used in formulas as building blocks. See | ||||
| # the section "Including formulas" for details. | ||||
| @@ -1814,8 +1739,8 @@ MATHJAX_RELPATH        = https://cdn.jsdelivr.net/npm/mathjax@2 | ||||
|  | ||||
| # The MATHJAX_EXTENSIONS tag can be used to specify one or more MathJax | ||||
| # extension names that should be enabled during MathJax rendering. For example | ||||
| # for MathJax version 2 (see | ||||
| # https://docs.mathjax.org/en/v2.7-latest/tex.html#tex-and-latex-extensions): | ||||
| # for MathJax version 2 (see https://docs.mathjax.org/en/v2.7-latest/tex.html | ||||
| # #tex-and-latex-extensions): | ||||
| # MATHJAX_EXTENSIONS = TeX/AMSmath TeX/AMSsymbols | ||||
| # For example for MathJax version 3 (see | ||||
| # http://docs.mathjax.org/en/latest/input/tex/extensions/index.html): | ||||
| @@ -2066,16 +1991,9 @@ PDF_HYPERLINKS         = YES | ||||
|  | ||||
| USE_PDFLATEX           = YES | ||||
|  | ||||
| # The LATEX_BATCHMODE tag signals the behavior of LaTeX in case of an error. | ||||
| # Possible values are: NO same as ERROR_STOP, YES same as BATCH, BATCH In batch | ||||
| # mode nothing is printed on the terminal, errors are scrolled as if <return> is | ||||
| # hit at every error; missing files that TeX tries to input or request from | ||||
| # keyboard input (\read on a not open input stream) cause the job to abort, | ||||
| # NON_STOP In nonstop mode the diagnostic message will appear on the terminal, | ||||
| # but there is no possibility of user interaction just like in batch mode, | ||||
| # SCROLL In scroll mode, TeX will stop only for missing files to input or if | ||||
| # keyboard input is necessary and ERROR_STOP In errorstop mode, TeX will stop at | ||||
| # each error, asking for user intervention. | ||||
| # If the LATEX_BATCHMODE tag is set to YES, doxygen will add the \batchmode | ||||
| # command to the generated LaTeX files. This will instruct LaTeX to keep running | ||||
| # if errors occur, instead of asking the user for help. | ||||
| # The default value is: NO. | ||||
| # This tag requires that the tag GENERATE_LATEX is set to YES. | ||||
|  | ||||
| @@ -2096,6 +2014,14 @@ LATEX_HIDE_INDICES     = NO | ||||
|  | ||||
| LATEX_BIB_STYLE        = plain | ||||
|  | ||||
| # If the LATEX_TIMESTAMP tag is set to YES then the footer of each generated | ||||
| # page will contain the date and time when the page was generated. Setting this | ||||
| # to NO can help when comparing the output of multiple runs. | ||||
| # The default value is: NO. | ||||
| # This tag requires that the tag GENERATE_LATEX is set to YES. | ||||
|  | ||||
| #LATEX_TIMESTAMP        = NO | ||||
|  | ||||
| # The LATEX_EMOJI_DIRECTORY tag is used to specify the (relative or absolute) | ||||
| # path from which the emoji images will be read. If a relative path is entered, | ||||
| # it will be relative to the LATEX_OUTPUT directory. If left blank the | ||||
| @@ -2261,39 +2187,13 @@ DOCBOOK_OUTPUT         = docbook | ||||
| #--------------------------------------------------------------------------- | ||||
|  | ||||
| # If the GENERATE_AUTOGEN_DEF tag is set to YES, doxygen will generate an | ||||
| # AutoGen Definitions (see https://autogen.sourceforge.net/) file that captures | ||||
| # AutoGen Definitions (see http://autogen.sourceforge.net/) file that captures | ||||
| # the structure of the code including all documentation. Note that this feature | ||||
| # is still experimental and incomplete at the moment. | ||||
| # The default value is: NO. | ||||
|  | ||||
| GENERATE_AUTOGEN_DEF   = NO | ||||
|  | ||||
| #--------------------------------------------------------------------------- | ||||
| # Configuration options related to Sqlite3 output | ||||
| #--------------------------------------------------------------------------- | ||||
|  | ||||
| # If the GENERATE_SQLITE3 tag is set to YES doxygen will generate a Sqlite3 | ||||
| # database with symbols found by doxygen stored in tables. | ||||
| # The default value is: NO. | ||||
|  | ||||
| #GENERATE_SQLITE3       = NO | ||||
|  | ||||
| # The SQLITE3_OUTPUT tag is used to specify where the Sqlite3 database will be | ||||
| # put. If a relative path is entered the value of OUTPUT_DIRECTORY will be put | ||||
| # in front of it. | ||||
| # The default directory is: sqlite3. | ||||
| # This tag requires that the tag GENERATE_SQLITE3 is set to YES. | ||||
|  | ||||
| #SQLITE3_OUTPUT         = sqlite3 | ||||
|  | ||||
| # The SQLITE3_OVERWRITE_DB tag is set to YES, the existing doxygen_sqlite3.db | ||||
| # database file will be recreated with each doxygen run. If set to NO, doxygen | ||||
| # will warn if an a database file is already found and not modify it. | ||||
| # The default value is: YES. | ||||
| # This tag requires that the tag GENERATE_SQLITE3 is set to YES. | ||||
|  | ||||
| #SQLITE3_RECREATE_DB    = YES | ||||
|  | ||||
| #--------------------------------------------------------------------------- | ||||
| # Configuration options related to the Perl module output | ||||
| #--------------------------------------------------------------------------- | ||||
| @@ -2436,15 +2336,15 @@ TAGFILES               = | ||||
|  | ||||
| GENERATE_TAGFILE       = | ||||
|  | ||||
| # If the ALLEXTERNALS tag is set to YES, all external classes and namespaces | ||||
| # will be listed in the class and namespace index. If set to NO, only the | ||||
| # inherited external classes will be listed. | ||||
| # If the ALLEXTERNALS tag is set to YES, all external class will be listed in | ||||
| # the class index. If set to NO, only the inherited external classes will be | ||||
| # listed. | ||||
| # The default value is: NO. | ||||
|  | ||||
| ALLEXTERNALS           = NO | ||||
|  | ||||
| # If the EXTERNAL_GROUPS tag is set to YES, all external groups will be listed | ||||
| # in the topic index. If set to NO, only the current project's groups will be | ||||
| # in the modules index. If set to NO, only the current project's groups will be | ||||
| # listed. | ||||
| # The default value is: YES. | ||||
|  | ||||
| @@ -2458,9 +2358,16 @@ EXTERNAL_GROUPS        = YES | ||||
| EXTERNAL_PAGES         = YES | ||||
|  | ||||
| #--------------------------------------------------------------------------- | ||||
| # Configuration options related to diagram generator tools | ||||
| # Configuration options related to the dot tool | ||||
| #--------------------------------------------------------------------------- | ||||
|  | ||||
| # You can include diagrams made with dia in doxygen documentation. Doxygen will | ||||
| # then run dia to produce the diagram and insert it in the documentation. The | ||||
| # DIA_PATH tag allows you to specify the directory where the dia binary resides. | ||||
| # If left empty dia is assumed to be found in the default search path. | ||||
|  | ||||
| DIA_PATH               = | ||||
|  | ||||
| # If set to YES the inheritance and collaboration graphs will hide inheritance | ||||
| # and usage relations if the target is undocumented or is not a class. | ||||
| # The default value is: YES. | ||||
| @@ -2469,10 +2376,10 @@ HIDE_UNDOC_RELATIONS   = YES | ||||
|  | ||||
| # If you set the HAVE_DOT tag to YES then doxygen will assume the dot tool is | ||||
| # available from the path. This tool is part of Graphviz (see: | ||||
| # https://www.graphviz.org/), a graph visualization toolkit from AT&T and Lucent | ||||
| # http://www.graphviz.org/), a graph visualization toolkit from AT&T and Lucent | ||||
| # Bell Labs. The other options in this section have no effect if this option is | ||||
| # set to NO | ||||
| # The default value is: YES. | ||||
| # The default value is: NO. | ||||
|  | ||||
| HAVE_DOT               = YES | ||||
|  | ||||
| @@ -2486,51 +2393,37 @@ HAVE_DOT               = YES | ||||
|  | ||||
| DOT_NUM_THREADS        = 0 | ||||
|  | ||||
| # DOT_COMMON_ATTR is common attributes for nodes, edges and labels of | ||||
| # subgraphs. When you want a differently looking font in the dot files that | ||||
| # doxygen generates you can specify fontname, fontcolor and fontsize attributes. | ||||
| # For details please see <a href=https://graphviz.org/doc/info/attrs.html>Node, | ||||
| # Edge and Graph Attributes specification</a> You need to make sure dot is able | ||||
| # to find the font, which can be done by putting it in a standard location or by | ||||
| # setting the DOTFONTPATH environment variable or by setting DOT_FONTPATH to the | ||||
| # directory containing the font. Default graphviz fontsize is 14. | ||||
| # The default value is: fontname=Helvetica,fontsize=10. | ||||
| # When you want a differently looking font in the dot files that doxygen | ||||
| # generates you can specify the font name using DOT_FONTNAME. You need to make | ||||
| # sure dot is able to find the font, which can be done by putting it in a | ||||
| # standard location or by setting the DOTFONTPATH environment variable or by | ||||
| # setting DOT_FONTPATH to the directory containing the font. | ||||
| # The default value is: Helvetica. | ||||
| # This tag requires that the tag HAVE_DOT is set to YES. | ||||
|  | ||||
| DOT_COMMON_ATTR        = "fontname=Helvetica,fontsize=10" | ||||
| #DOT_FONTNAME           = Helvetica | ||||
|  | ||||
| # DOT_EDGE_ATTR is concatenated with DOT_COMMON_ATTR. For elegant style you can | ||||
| # add 'arrowhead=open, arrowtail=open, arrowsize=0.5'. <a | ||||
| # href=https://graphviz.org/doc/info/arrows.html>Complete documentation about | ||||
| # arrows shapes.</a> | ||||
| # The default value is: labelfontname=Helvetica,labelfontsize=10. | ||||
| # The DOT_FONTSIZE tag can be used to set the size (in points) of the font of | ||||
| # dot graphs. | ||||
| # Minimum value: 4, maximum value: 24, default value: 10. | ||||
| # This tag requires that the tag HAVE_DOT is set to YES. | ||||
|  | ||||
| DOT_EDGE_ATTR          = "labelfontname=Helvetica,labelfontsize=10" | ||||
| #DOT_FONTSIZE           = 10 | ||||
|  | ||||
| # DOT_NODE_ATTR is concatenated with DOT_COMMON_ATTR. For view without boxes | ||||
| # around nodes set 'shape=plain' or 'shape=plaintext' <a | ||||
| # href=https://www.graphviz.org/doc/info/shapes.html>Shapes specification</a> | ||||
| # The default value is: shape=box,height=0.2,width=0.4. | ||||
| # By default doxygen will tell dot to use the default font as specified with | ||||
| # DOT_FONTNAME. If you specify a different font using DOT_FONTNAME you can set | ||||
| # the path where dot can find it using this tag. | ||||
| # This tag requires that the tag HAVE_DOT is set to YES. | ||||
|  | ||||
| DOT_NODE_ATTR          = "shape=box,height=0.2,width=0.4" | ||||
| #DOT_FONTPATH           = | ||||
|  | ||||
| # You can set the path where dot can find font specified with fontname in | ||||
| # DOT_COMMON_ATTR and others dot attributes. | ||||
| # This tag requires that the tag HAVE_DOT is set to YES. | ||||
|  | ||||
| DOT_FONTPATH           = | ||||
|  | ||||
| # If the CLASS_GRAPH tag is set to YES or GRAPH or BUILTIN then doxygen will | ||||
| # generate a graph for each documented class showing the direct and indirect | ||||
| # inheritance relations. In case the CLASS_GRAPH tag is set to YES or GRAPH and | ||||
| # HAVE_DOT is enabled as well, then dot will be used to draw the graph. In case | ||||
| # the CLASS_GRAPH tag is set to YES and HAVE_DOT is disabled or if the | ||||
| # CLASS_GRAPH tag is set to BUILTIN, then the built-in generator will be used. | ||||
| # If the CLASS_GRAPH tag is set to TEXT the direct and indirect inheritance | ||||
| # relations will be shown as texts / links. | ||||
| # Possible values are: NO, YES, TEXT, GRAPH and BUILTIN. | ||||
| # If the CLASS_GRAPH tag is set to YES (or GRAPH) then doxygen will generate a | ||||
| # graph for each documented class showing the direct and indirect inheritance | ||||
| # relations. In case HAVE_DOT is set as well dot will be used to draw the graph, | ||||
| # otherwise the built-in generator will be used. If the CLASS_GRAPH tag is set | ||||
| # to TEXT the direct and indirect inheritance relations will be shown as texts / | ||||
| # links. | ||||
| # Possible values are: NO, YES, TEXT and GRAPH. | ||||
| # The default value is: YES. | ||||
|  | ||||
| CLASS_GRAPH            = YES | ||||
| @@ -2538,21 +2431,15 @@ CLASS_GRAPH            = YES | ||||
| # If the COLLABORATION_GRAPH tag is set to YES then doxygen will generate a | ||||
| # graph for each documented class showing the direct and indirect implementation | ||||
| # dependencies (inheritance, containment, and class references variables) of the | ||||
| # class with other documented classes. Explicit enabling a collaboration graph, | ||||
| # when COLLABORATION_GRAPH is set to NO, can be accomplished by means of the | ||||
| # command \collaborationgraph. Disabling a collaboration graph can be | ||||
| # accomplished by means of the command \hidecollaborationgraph. | ||||
| # class with other documented classes. | ||||
| # The default value is: YES. | ||||
| # This tag requires that the tag HAVE_DOT is set to YES. | ||||
|  | ||||
| COLLABORATION_GRAPH    = YES | ||||
|  | ||||
| # If the GROUP_GRAPHS tag is set to YES then doxygen will generate a graph for | ||||
| # groups, showing the direct groups dependencies. Explicit enabling a group | ||||
| # dependency graph, when GROUP_GRAPHS is set to NO, can be accomplished by means | ||||
| # of the command \groupgraph. Disabling a directory graph can be accomplished by | ||||
| # means of the command \hidegroupgraph. See also the chapter Grouping in the | ||||
| # manual. | ||||
| # groups, showing the direct groups dependencies. See also the chapter Grouping | ||||
| # in the manual. | ||||
| # The default value is: YES. | ||||
| # This tag requires that the tag HAVE_DOT is set to YES. | ||||
|  | ||||
| @@ -2612,9 +2499,7 @@ TEMPLATE_RELATIONS     = NO | ||||
| # If the INCLUDE_GRAPH, ENABLE_PREPROCESSING and SEARCH_INCLUDES tags are set to | ||||
| # YES then doxygen will generate a graph for each documented file showing the | ||||
| # direct and indirect include dependencies of the file with other documented | ||||
| # files. Explicit enabling an include graph, when INCLUDE_GRAPH is is set to NO, | ||||
| # can be accomplished by means of the command \includegraph. Disabling an | ||||
| # include graph can be accomplished by means of the command \hideincludegraph. | ||||
| # files. | ||||
| # The default value is: YES. | ||||
| # This tag requires that the tag HAVE_DOT is set to YES. | ||||
|  | ||||
| @@ -2623,10 +2508,7 @@ INCLUDE_GRAPH          = YES | ||||
| # If the INCLUDED_BY_GRAPH, ENABLE_PREPROCESSING and SEARCH_INCLUDES tags are | ||||
| # set to YES then doxygen will generate a graph for each documented file showing | ||||
| # the direct and indirect include dependencies of the file with other documented | ||||
| # files. Explicit enabling an included by graph, when INCLUDED_BY_GRAPH is set | ||||
| # to NO, can be accomplished by means of the command \includedbygraph. Disabling | ||||
| # an included by graph can be accomplished by means of the command | ||||
| # \hideincludedbygraph. | ||||
| # files. | ||||
| # The default value is: YES. | ||||
| # This tag requires that the tag HAVE_DOT is set to YES. | ||||
|  | ||||
| @@ -2666,10 +2548,7 @@ GRAPHICAL_HIERARCHY    = YES | ||||
| # If the DIRECTORY_GRAPH tag is set to YES then doxygen will show the | ||||
| # dependencies a directory has on other directories in a graphical way. The | ||||
| # dependency relations are determined by the #include relations between the | ||||
| # files in the directories. Explicit enabling a directory graph, when | ||||
| # DIRECTORY_GRAPH is set to NO, can be accomplished by means of the command | ||||
| # \directorygraph. Disabling a directory graph can be accomplished by means of | ||||
| # the command \hidedirectorygraph. | ||||
| # files in the directories. | ||||
| # The default value is: YES. | ||||
| # This tag requires that the tag HAVE_DOT is set to YES. | ||||
|  | ||||
| @@ -2685,13 +2564,12 @@ DIR_GRAPH_MAX_DEPTH    = 1 | ||||
| # The DOT_IMAGE_FORMAT tag can be used to set the image format of the images | ||||
| # generated by dot. For an explanation of the image formats see the section | ||||
| # output formats in the documentation of the dot tool (Graphviz (see: | ||||
| # https://www.graphviz.org/)). | ||||
| # http://www.graphviz.org/)). | ||||
| # Note: If you choose svg you need to set HTML_FILE_EXTENSION to xhtml in order | ||||
| # to make the SVG files visible in IE 9+ (other browsers do not have this | ||||
| # requirement). | ||||
| # Possible values are: png, jpg, jpg:cairo, jpg:cairo:gd, jpg:gd, jpg:gd:gd, | ||||
| # gif, gif:cairo, gif:cairo:gd, gif:gd, gif:gd:gd, svg, png:gd, png:gd:gd, | ||||
| # png:cairo, png:cairo:gd, png:cairo:cairo, png:cairo:gdiplus, png:gdiplus and | ||||
| # Possible values are: png, jpg, gif, svg, png:gd, png:gd:gd, png:cairo, | ||||
| # png:cairo:gd, png:cairo:cairo, png:cairo:gdiplus, png:gdiplus and | ||||
| # png:gdiplus:gdiplus. | ||||
| # The default value is: png. | ||||
| # This tag requires that the tag HAVE_DOT is set to YES. | ||||
| @@ -2723,12 +2601,11 @@ DOT_PATH               = | ||||
|  | ||||
| DOTFILE_DIRS           = | ||||
|  | ||||
| # You can include diagrams made with dia in doxygen documentation. Doxygen will | ||||
| # then run dia to produce the diagram and insert it in the documentation. The | ||||
| # DIA_PATH tag allows you to specify the directory where the dia binary resides. | ||||
| # If left empty dia is assumed to be found in the default search path. | ||||
| # The MSCFILE_DIRS tag can be used to specify one or more directories that | ||||
| # contain msc files that are included in the documentation (see the \mscfile | ||||
| # command). | ||||
|  | ||||
| DIA_PATH               = | ||||
| MSCFILE_DIRS           = | ||||
|  | ||||
| # The DIAFILE_DIRS tag can be used to specify one or more directories that | ||||
| # contain dia files that are included in the documentation (see the \diafile | ||||
| @@ -2778,6 +2655,18 @@ DOT_GRAPH_MAX_NODES    = 50 | ||||
|  | ||||
| MAX_DOT_GRAPH_DEPTH    = 0 | ||||
|  | ||||
| # Set the DOT_TRANSPARENT tag to YES to generate images with a transparent | ||||
| # background. This is disabled by default, because dot on Windows does not seem | ||||
| # to support this out of the box. | ||||
| # | ||||
| # Warning: Depending on the platform used, enabling this option may lead to | ||||
| # badly anti-aliased labels on the edges of a graph (i.e. they become hard to | ||||
| # read). | ||||
| # The default value is: NO. | ||||
| # This tag requires that the tag HAVE_DOT is set to YES. | ||||
|  | ||||
| #DOT_TRANSPARENT        = NO | ||||
|  | ||||
| # Set the DOT_MULTI_TARGETS tag to YES to allow dot to generate multiple output | ||||
| # files in one run (i.e. multiple -o and -T options on the command line). This | ||||
| # makes dot run faster, but since only newer versions of dot (>1.8.10) support | ||||
| @@ -2805,19 +2694,3 @@ GENERATE_LEGEND        = YES | ||||
| # The default value is: YES. | ||||
|  | ||||
| DOT_CLEANUP            = YES | ||||
|  | ||||
| # You can define message sequence charts within doxygen comments using the \msc | ||||
| # command. If the MSCGEN_TOOL tag is left empty (the default), then doxygen will | ||||
| # use a built-in version of mscgen tool to produce the charts. Alternatively, | ||||
| # the MSCGEN_TOOL tag can also specify the name an external tool. For instance, | ||||
| # specifying prog as the value, doxygen will call the tool as prog -T | ||||
| # <outfile_format> -o <outputfile> <inputfile>. The external tool should support | ||||
| # output file formats "png", "eps", "svg", and "ismap". | ||||
|  | ||||
| MSCGEN_TOOL            = | ||||
|  | ||||
| # The MSCFILE_DIRS tag can be used to specify one or more directories that | ||||
| # contain msc files that are included in the documentation (see the \mscfile | ||||
| # command). | ||||
|  | ||||
| MSCFILE_DIRS           = | ||||
|   | ||||
							
								
								
									
										11
									
								
								GNUmakefile
									
									
									
									
									
								
							
							
						
						
									
										11
									
								
								GNUmakefile
									
									
									
									
									
								
							| @@ -16,9 +16,9 @@ MAKEFLAGS += --no-builtin-rules | ||||
| ## LD := Linker. | ||||
| ## ANDROID_SDK := Path to the Android SDK. | ||||
|  | ||||
| VERSION_CODE := 41 | ||||
| VERSION_CODE_IOS := 16 | ||||
| VERSION_NUMBER := 0.2025.8-wip | ||||
| VERSION_CODE := 43 | ||||
| VERSION_CODE_IOS := 17 | ||||
| VERSION_NUMBER := 0.2025.9 | ||||
| VERSION_NAME := This program kills fascists. | ||||
|  | ||||
| IPHONEOS_VERSION_MIN=14.0 | ||||
| @@ -253,7 +253,10 @@ $(ANDROID_TARGETS): CFLAGS += \ | ||||
| 	-fno-asynchronous-unwind-tables \ | ||||
| 	-funwind-tables \ | ||||
| 	-Wno-unknown-warning-option | ||||
| $(ANDROID_TARGETS): LDFLAGS += --sysroot $(ANDROID_NDK)/toolchains/llvm/prebuilt/linux-x86_64/sysroot -fPIC | ||||
| $(ANDROID_TARGETS): LDFLAGS += \ | ||||
| 	--sysroot $(ANDROID_NDK)/toolchains/llvm/prebuilt/linux-x86_64/sysroot \ | ||||
| 	-Wl,-z,max-page-size=16384 \ | ||||
| 	-fPIC | ||||
| $(DEBUG_TARGETS): CFLAGS += -DDEBUG -Og | ||||
| $(DEBUG_TARGETS): LDFLAGS += -Og | ||||
| $(RELEASE_TARGETS): CFLAGS += \ | ||||
|   | ||||
| @@ -1,5 +1,5 @@ | ||||
| { | ||||
| 	"type": "tildefriends-app", | ||||
| 	"emoji": "🦀", | ||||
| 	"previous": "&TTGzyovmfKozjELCGPBFLLEXQcpfaArOMmqzemvz9J8=.sha256" | ||||
| 	"previous": "&IDzjVQjtPyhesUrl45qkZFjzWl0xVlj+2M/XXQRvXO0=.sha256" | ||||
| } | ||||
|   | ||||
| @@ -76,6 +76,9 @@ tfrpc.register(function setHash(hash) { | ||||
| core.register('onMessage', async function (id) { | ||||
| 	await tfrpc.rpc.notifyNewMessage(id); | ||||
| }); | ||||
| core.register('onBlob', async function (id) { | ||||
| 	await tfrpc.rpc.notifyNewBlob(id); | ||||
| }); | ||||
| tfrpc.register(async function store_blob(blob) { | ||||
| 	if (Array.isArray(blob)) { | ||||
| 		blob = Uint8Array.from(blob); | ||||
|   | ||||
| @@ -21,7 +21,9 @@ class TfElement extends LitElement { | ||||
| 			channels_latest: {type: Object}, | ||||
| 			guest: {type: Boolean}, | ||||
| 			url: {type: String}, | ||||
| 			private_closed: {type: Object}, | ||||
| 			private_messages: {type: Array}, | ||||
| 			grouped_private_messages: {type: Object}, | ||||
| 			recent_reactions: {type: Array}, | ||||
| 			is_administrator: {type: Boolean}, | ||||
| 			stay_connected: {type: Boolean}, | ||||
| @@ -48,6 +50,7 @@ class TfElement extends LitElement { | ||||
| 		this.loading_latest = 0; | ||||
| 		this.loading_latest_scheduled = 0; | ||||
| 		this.recent_reactions = []; | ||||
| 		this.private_closed = {}; | ||||
| 		tfrpc.rpc.getBroadcasts().then((b) => { | ||||
| 			self.broadcasts = b || []; | ||||
| 		}); | ||||
| @@ -62,6 +65,17 @@ class TfElement extends LitElement { | ||||
| 		tfrpc.register(async function notifyNewMessage(id) { | ||||
| 			await self.fetch_new_message(id); | ||||
| 		}); | ||||
| 		tfrpc.register(async function notifyNewBlob(id) { | ||||
| 			window.dispatchEvent( | ||||
| 				new CustomEvent('blob-stored', { | ||||
| 					bubbles: true, | ||||
| 					composed: true, | ||||
| 					detail: { | ||||
| 						id: id, | ||||
| 					}, | ||||
| 				}) | ||||
| 			); | ||||
| 		}); | ||||
| 		tfrpc.register(function set(name, value) { | ||||
| 			if (name === 'broadcasts') { | ||||
| 				self.broadcasts = value; | ||||
| @@ -85,9 +99,22 @@ class TfElement extends LitElement { | ||||
| 		this.whoami = whoami ?? (ids.length ? ids[0] : undefined); | ||||
| 		this.guest = !this.whoami?.length; | ||||
| 		this.ids = ids; | ||||
| 		let private_closed = | ||||
| 			(await tfrpc.rpc.databaseGet('private_closed')) ?? '{}'; | ||||
| 		this.private_closed = JSON.parse(private_closed); | ||||
| 		await this.load_channels(); | ||||
| 	} | ||||
|  | ||||
| 	async close_private_chat(event) { | ||||
| 		let update = {}; | ||||
| 		update[event.detail.key] = true; | ||||
| 		this.private_closed = Object.assign(update, this.private_closed); | ||||
| 		await tfrpc.rpc.databaseSet( | ||||
| 			'private_closed', | ||||
| 			JSON.stringify(this.private_closed) | ||||
| 		); | ||||
| 	} | ||||
|  | ||||
| 	async load_channels() { | ||||
| 		let channels = await tfrpc.rpc.query( | ||||
| 			` | ||||
| @@ -135,12 +162,32 @@ class TfElement extends LitElement { | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	visible_private() { | ||||
| 		if (!this.grouped_private_messages || !this.private_closed) { | ||||
| 			return []; | ||||
| 		} | ||||
| 		let self = this; | ||||
| 		return Object.fromEntries( | ||||
| 			Object.entries(this.grouped_private_messages).filter(([key, value]) => { | ||||
| 				let channel = '🔐' + [...new Set(JSON.parse(key))].sort().join(','); | ||||
| 				let grouped_latest = Math.max(...value.map((x) => x.rowid)); | ||||
| 				return ( | ||||
| 					!self.private_closed[key] || | ||||
| 					self.channels_unread[channel] === undefined || | ||||
| 					grouped_latest > self.channels_unread[channel] | ||||
| 				); | ||||
| 			}) | ||||
| 		); | ||||
| 	} | ||||
|  | ||||
| 	next_channel(delta) { | ||||
| 		let channel_names = [ | ||||
| 			'', | ||||
| 			'@', | ||||
| 			'👍', | ||||
| 			'🔐', | ||||
| 			...Object.keys(this.visible_private()) | ||||
| 				.sort() | ||||
| 				.map((x) => '🔐' + JSON.parse(x).join(',')), | ||||
| 			...this.channels.map((x) => '#' + x), | ||||
| 		]; | ||||
| 		let index = channel_names.indexOf(this.hash.substring(1)); | ||||
| @@ -366,6 +413,36 @@ class TfElement extends LitElement { | ||||
| 		return result; | ||||
| 	} | ||||
|  | ||||
| 	async group_private_messages(messages) { | ||||
| 		let groups = {}; | ||||
| 		let result = await this.decrypt( | ||||
| 			await tfrpc.rpc.query( | ||||
| 				` | ||||
| 				SELECT messages.rowid, messages.id, author, timestamp, json(content) AS content | ||||
| 				FROM messages | ||||
| 				JOIN json_each(?) AS ids | ||||
| 				WHERE messages.id = ids.value | ||||
| 				ORDER BY timestamp DESC | ||||
| 			`, | ||||
| 				[JSON.stringify(messages)] | ||||
| 			) | ||||
| 		); | ||||
| 		for (let message of result) { | ||||
| 			let key = JSON.stringify( | ||||
| 				[ | ||||
| 					...new Set( | ||||
| 						message?.decrypted?.recps?.filter((x) => x != this.whoami) | ||||
| 					), | ||||
| 				].sort() ?? [] | ||||
| 			); | ||||
| 			if (!groups[key]) { | ||||
| 				groups[key] = []; | ||||
| 			} | ||||
| 			groups[key].push(message); | ||||
| 		} | ||||
| 		return groups; | ||||
| 	} | ||||
|  | ||||
| 	async load_channels_latest(following) { | ||||
| 		let start_time = new Date(); | ||||
| 		let latest_private = this.get_latest_private(following); | ||||
| @@ -438,12 +515,15 @@ class TfElement extends LitElement { | ||||
| 		console.log('channels took', (new Date() - start_time) / 1000.0); | ||||
| 		let self = this; | ||||
| 		start_time = new Date(); | ||||
| 		latest_private.then(function (latest) { | ||||
| 		latest_private.then(async function (latest) { | ||||
| 			self.channels_latest = Object.assign({}, self.channels_latest, { | ||||
| 				'🔐': latest[0], | ||||
| 			}); | ||||
| 			console.log('private took', (new Date() - start_time) / 1000.0); | ||||
| 			self.private_messages = latest[1]; | ||||
| 			self.grouped_private_messages = await self.group_private_messages( | ||||
| 				latest[1] | ||||
| 			); | ||||
| 		}); | ||||
| 	} | ||||
|  | ||||
| @@ -628,8 +708,10 @@ class TfElement extends LitElement { | ||||
| 					@refresh=${this.refresh} | ||||
| 					@toggle_stay_connected=${this.toggle_stay_connected} | ||||
| 					@loadmessages=${this.reset_progress} | ||||
| 					@closeprivatechat=${this.close_private_chat} | ||||
| 					.connections=${this.connections} | ||||
| 					.private_messages=${this.private_messages} | ||||
| 					.grouped_private_messages=${this.visible_private()} | ||||
| 					.recent_reactions=${this.recent_reactions} | ||||
| 					?is_administrator=${this.is_administrator} | ||||
| 					?stay_connected=${this.stay_connected} | ||||
| @@ -661,7 +743,7 @@ class TfElement extends LitElement { | ||||
| 					whoami=${this.whoami} | ||||
| 					.users=${this.users} | ||||
| 					query=${this.hash?.startsWith('#sql=') | ||||
| 						? decodeURIComponent(this.hash.substring(5)) | ||||
| 						? this.hash.substring(5) | ||||
| 						: null} | ||||
| 				></tf-tab-query> | ||||
| 			`; | ||||
| @@ -718,14 +800,13 @@ class TfElement extends LitElement { | ||||
| 				class="w3-bar w3-theme-l1" | ||||
| 				style="position: static; top: 0; z-index: 10" | ||||
| 			> | ||||
| 				${this.is_administrator && self.tab != 'news' | ||||
| 				${this.is_administrator | ||||
| 					? html` | ||||
| 							<button | ||||
| 								class=${'w3-bar-item w3-button w3-circle w3-ripple' + | ||||
| 								(this.connections?.some((x) => x.flags.one_shot) | ||||
| 									? ' w3-spin' | ||||
| 									: '')} | ||||
| 								style="width: 1.5em; height: 1.5em; padding: 8px" | ||||
| 								@click=${this.refresh} | ||||
| 							> | ||||
| 								↻ | ||||
|   | ||||
| @@ -16,6 +16,7 @@ class TfComposeElement extends LitElement { | ||||
| 			author: {type: String}, | ||||
| 			channel: {type: String}, | ||||
| 			new_thread: {type: Boolean}, | ||||
| 			recipients: {type: Array}, | ||||
| 		}; | ||||
| 	} | ||||
|  | ||||
| @@ -91,7 +92,9 @@ class TfComposeElement extends LitElement { | ||||
| 				bubbles: true, | ||||
| 				composed: true, | ||||
| 				detail: { | ||||
| 					id: this.branch, | ||||
| 					id: | ||||
| 						this.branch ?? | ||||
| 						(this.recipients ? this.recipients.join(',') : undefined), | ||||
| 					draft: draft, | ||||
| 				}, | ||||
| 			}) | ||||
| @@ -291,7 +294,7 @@ class TfComposeElement extends LitElement { | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	firstUpdated() { | ||||
| 	get_values() { | ||||
| 		let values = Object.entries(this.users).map((x) => ({ | ||||
| 			key: x[1].name ?? x[0], | ||||
| 			value: x[0], | ||||
| @@ -307,11 +310,15 @@ class TfComposeElement extends LitElement { | ||||
| 				values | ||||
| 			); | ||||
| 		} | ||||
| 		return values; | ||||
| 	} | ||||
|  | ||||
| 	firstUpdated() { | ||||
| 		let tribute = new Tribute({ | ||||
| 			iframe: this.shadowRoot, | ||||
| 			collection: [ | ||||
| 				{ | ||||
| 					values: values, | ||||
| 					values: this.get_values(), | ||||
| 					selectTemplate: function (item) { | ||||
| 						return item | ||||
| 							? `[@${item.original.key}](${item.original.value})` | ||||
| @@ -330,6 +337,7 @@ class TfComposeElement extends LitElement { | ||||
| 			], | ||||
| 		}); | ||||
| 		tribute.attach(this.renderRoot.getElementById('edit')); | ||||
| 		this._tribute = tribute; | ||||
| 	} | ||||
|  | ||||
| 	updated() { | ||||
| @@ -340,6 +348,7 @@ class TfComposeElement extends LitElement { | ||||
| 			preview.innerHTML = this.process_text(edit.innerText); | ||||
| 			this.last_updated_text = edit.innerText; | ||||
| 		} | ||||
| 		this._tribute.collection[0].values = this.get_values(); | ||||
| 		let encrypt = this.renderRoot.getElementById('encrypt_to'); | ||||
| 		if (encrypt) { | ||||
| 			let tribute = new Tribute({ | ||||
| @@ -496,7 +505,17 @@ class TfComposeElement extends LitElement { | ||||
| 	} | ||||
|  | ||||
| 	get_draft() { | ||||
| 		return this.drafts[this.branch || ''] || {}; | ||||
| 		let key = | ||||
| 			this.branch || | ||||
| 			(this.recipients ? this.recipients.join(',') : undefined) || | ||||
| 			''; | ||||
| 		let draft = this.drafts[key] || {}; | ||||
| 		if (this.recipients && !draft.encrypt_to?.length) { | ||||
| 			draft.encrypt_to = [ | ||||
| 				...new Set(this.recipients).union(new Set(draft.encrypt_to ?? [])), | ||||
| 			]; | ||||
| 		} | ||||
| 		return draft; | ||||
| 	} | ||||
|  | ||||
| 	update_encrypt(event) { | ||||
| @@ -606,7 +625,7 @@ class TfComposeElement extends LitElement { | ||||
| 					<div class="w3-half"> | ||||
| 						<span | ||||
| 							class="w3-input w3-theme-d1 w3-border" | ||||
| 							style="resize: vertical; width: 100%; overflow: hidden; white-space: pre-wrap" | ||||
| 							style="resize: vertical; width: 100%; white-space: pre-wrap" | ||||
| 							placeholder="Write a post here." | ||||
| 							id="edit" | ||||
| 							@input=${this.input} | ||||
|   | ||||
| @@ -45,11 +45,14 @@ class TfMessageElement extends LitElement { | ||||
| 	connectedCallback() { | ||||
| 		super.connectedCallback(); | ||||
| 		this._click_callback = this.document_click.bind(this); | ||||
| 		this._blob_stored = this.blob_stored.bind(this); | ||||
| 		document.body.addEventListener('mouseup', this._click_callback); | ||||
| 		window.addEventListener('blob-stored', this._blob_stored); | ||||
| 	} | ||||
|  | ||||
| 	disconnectedCallback() { | ||||
| 		super.disconnectedCallback(); | ||||
| 		window.removeEventListener('blob-stored', this._blob_stored); | ||||
| 		document.body.removeEventListener('mouseup', this._click_callback); | ||||
| 	} | ||||
|  | ||||
| @@ -61,6 +64,16 @@ class TfMessageElement extends LitElement { | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	blob_stored(event) { | ||||
| 		let search = `/${event.detail.id}/view`; | ||||
| 		for (let img of this.shadowRoot.querySelectorAll('img')) { | ||||
| 			if (img.src.indexOf(search) != -1) { | ||||
| 				let src = img.src.split('?')[0]; | ||||
| 				img.src = `${src}?${new Date().valueOf()}`; | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	show_reply() { | ||||
| 		let event = new CustomEvent('tf-draft', { | ||||
| 			bubbles: true, | ||||
| @@ -537,7 +550,7 @@ class TfMessageElement extends LitElement { | ||||
| 			</style> | ||||
| 			<div | ||||
| 				class="w3-card-4 ${this.class_background()} w3-border-theme w3-margin-top" | ||||
| 				style="overflow: auto; overflow-wrap: anywhere; display: block; max-width: 100%" | ||||
| 				style="overflow-wrap: anywhere; display: block; max-width: 100%" | ||||
| 			> | ||||
| 				${inner} | ||||
| 			</div> | ||||
|   | ||||
| @@ -349,6 +349,9 @@ class TfProfileElement extends LitElement { | ||||
| 			${until(this.load_follows(), html`<p>Loading accounts followed...</p>`)} | ||||
| 			<footer class="w3-container"> | ||||
| 				<p> | ||||
| 					<a class="w3-button w3-theme-d1" href=${'#🔐' + (this.id != this.whoami ? this.id : '')}> | ||||
| 						Open Private Chat | ||||
| 					</a> | ||||
| 					${edit} | ||||
| 					${follow} | ||||
| 					${block} | ||||
|   | ||||
| @@ -43,6 +43,8 @@ const tf = css` | ||||
| 		border-left: 4px solid #fff; | ||||
| 		padding: 8px; | ||||
| 		padding-left: 12px; | ||||
| 		margin-left: 0; | ||||
| 		margin-right: 0; | ||||
| 	} | ||||
| `; | ||||
|  | ||||
|   | ||||
| @@ -18,6 +18,7 @@ class TfTabNewsFeedElement extends LitElement { | ||||
| 			time_range: {type: Array}, | ||||
| 			time_loading: {type: Array}, | ||||
| 			private_messages: {type: Array}, | ||||
| 			grouped_private_messages: {type: Object}, | ||||
| 			recent_reactions: {type: Array}, | ||||
| 		}; | ||||
| 	} | ||||
| @@ -208,7 +209,9 @@ class TfTabNewsFeedElement extends LitElement { | ||||
| 			console.log( | ||||
| 				`load of ${result.length} rows took ${(t2 - t0) / 1000} (${(t1 - t0) / 1000} to find ${initial_messages.length} initial messages, ${(t2 - t1) / 1000} to find ${result.length} total messages) following=${this.following.length} st=${start_time} et=${end_time}` | ||||
| 			); | ||||
| 		} else if (this.hash == '#🔐') { | ||||
| 		} else if (this.hash.startsWith('#🔐')) { | ||||
| 			let ids = | ||||
| 				this.hash == '#🔐' ? [] : this.hash.substring('#🔐'.length).split(','); | ||||
| 			result = await tfrpc.rpc.query( | ||||
| 				` | ||||
| 					SELECT TRUE AS is_primary, messages.rowid, messages.id, previous, author, sequence, timestamp, hash, json(content) AS content, signature | ||||
| @@ -220,7 +223,11 @@ class TfTabNewsFeedElement extends LitElement { | ||||
| 					ORDER BY messages.rowid DESC LIMIT ?4 | ||||
| 				`, | ||||
| 				[ | ||||
| 					JSON.stringify(this.private_messages), | ||||
| 					JSON.stringify( | ||||
| 						this.grouped_private_messages?.[JSON.stringify(ids)]?.map( | ||||
| 							(x) => x.id | ||||
| 						) ?? [] | ||||
| 					), | ||||
| 					start_time, | ||||
| 					end_time, | ||||
| 					k_max_results, | ||||
| @@ -379,13 +386,16 @@ class TfTabNewsFeedElement extends LitElement { | ||||
| 		let self = this; | ||||
| 		this.loading++; | ||||
| 		let messages = []; | ||||
| 		let original_hash = this.hash; | ||||
| 		try { | ||||
| 			if (this._messages_hash !== this.hash) { | ||||
| 				this.messages = []; | ||||
| 				this._messages_hash = this.hash; | ||||
| 			} | ||||
| 			this._messages_following = JSON.stringify(this.following); | ||||
| 			this._private_messages = JSON.stringify(this.private_messages); | ||||
| 			this._private_messages = | ||||
| 				JSON.stringify(this.private_messages) + | ||||
| 				JSON.stringify(this.grouped_private_messages); | ||||
| 			let now = new Date().valueOf(); | ||||
| 			let start_time = now - 24 * 60 * 60 * 1000; | ||||
| 			this.start_time = start_time; | ||||
| @@ -398,7 +408,9 @@ class TfTabNewsFeedElement extends LitElement { | ||||
| 		} finally { | ||||
| 			this.loading--; | ||||
| 		} | ||||
| 		if (this.hash == original_hash) { | ||||
| 			this.messages = this.merge_messages(this.messages, messages); | ||||
| 		} | ||||
| 		this.time_loading = undefined; | ||||
| 		console.log( | ||||
| 			`loading ${messages.length} messages done for ${self.whoami} in ${(new Date() - start_time) / 1000}s` | ||||
| @@ -424,12 +436,42 @@ class TfTabNewsFeedElement extends LitElement { | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	close_private_chat() { | ||||
| 		this.mark_all_read(); | ||||
| 		this.dispatchEvent( | ||||
| 			new CustomEvent('closeprivatechat', { | ||||
| 				bubbles: true, | ||||
| 				composed: true, | ||||
| 				detail: { | ||||
| 					key: JSON.stringify( | ||||
| 						this.hash == '#🔐' | ||||
| 							? [] | ||||
| 							: this.hash.substring('#🔐'.length).split(',') | ||||
| 					), | ||||
| 				}, | ||||
| 			}) | ||||
| 		); | ||||
| 		tfrpc.rpc.setHash('#'); | ||||
| 	} | ||||
|  | ||||
| 	render_close_chat_button() { | ||||
| 		if (this.hash.startsWith('#🔐')) { | ||||
| 			return html` | ||||
| 				<button class="w3-button w3-theme-d1" @click=${this.close_private_chat}> | ||||
| 					Close Chat | ||||
| 				</button> | ||||
| 			`; | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	render() { | ||||
| 		if ( | ||||
| 			!this.messages || | ||||
| 			this._messages_hash !== this.hash || | ||||
| 			this._messages_following !== JSON.stringify(this.following) || | ||||
| 			this._private_messages !== JSON.stringify(this.private_messages) | ||||
| 			this._private_messages !== | ||||
| 				JSON.stringify(this.private_messages) + | ||||
| 					JSON.stringify(this.grouped_private_messages) | ||||
| 		) { | ||||
| 			console.log( | ||||
| 				`loading messages for ${this.whoami} (following ${this.following.length})` | ||||
| @@ -489,6 +531,7 @@ class TfTabNewsFeedElement extends LitElement { | ||||
| 						Mark All Read | ||||
| 					</button>` | ||||
| 				: undefined} | ||||
| 			${this.render_close_chat_button()} | ||||
| 			<tf-news | ||||
| 				id="news" | ||||
| 				whoami=${this.whoami} | ||||
|   | ||||
| @@ -24,6 +24,7 @@ class TfTabNewsElement extends LitElement { | ||||
| 			channels_latest: {type: Object}, | ||||
| 			connections: {type: Array}, | ||||
| 			private_messages: {type: Array}, | ||||
| 			grouped_private_messages: {type: Object}, | ||||
| 			recent_reactions: {type: Array}, | ||||
| 			peer_exchange: {type: Boolean}, | ||||
| 			is_administrator: {type: Boolean}, | ||||
| @@ -115,6 +116,19 @@ class TfTabNewsElement extends LitElement { | ||||
| 			) { | ||||
| 				return '✉️ '; | ||||
| 			} | ||||
| 		} else if (channel?.startsWith('🔐')) { | ||||
| 			let key = JSON.stringify(channel.substring('🔐'.length).split(',')); | ||||
| 			if (this.grouped_private_messages?.[key]) { | ||||
| 				let grouped_latest = Math.max( | ||||
| 					...this.grouped_private_messages?.[key]?.map((x) => x.rowid) | ||||
| 				); | ||||
| 				if ( | ||||
| 					this.channels_unread[channel] === undefined || | ||||
| 					grouped_latest > this.channels_unread[channel] | ||||
| 				) { | ||||
| 					return '✉️ '; | ||||
| 				} | ||||
| 			} | ||||
| 		} else if ( | ||||
| 			this.channels_latest[channel] && | ||||
| 			this.channels_latest[channel] > 0 && | ||||
| @@ -257,12 +271,29 @@ class TfTabNewsElement extends LitElement { | ||||
| 					style=${this.hash == '#👍' ? 'font-weight: bold' : undefined} | ||||
| 					>${this.unread_status('👍')}👍votes</a | ||||
| 				> | ||||
| 				${Object.keys(this?.grouped_private_messages ?? []) | ||||
| 					?.sort() | ||||
| 					?.map( | ||||
| 						(key) => html` | ||||
| 							<a | ||||
| 					href="#🔐" | ||||
| 								href=${'#🔐' + JSON.parse(key).join(',')} | ||||
| 								class="w3-bar-item w3-button" | ||||
| 					style=${this.hash == '#🔐' ? 'font-weight: bold' : undefined} | ||||
| 					>${this.unread_status('🔐')}🔐private</a | ||||
| 								style=${this.hash == '#🔐' + JSON.parse(key).join(',') | ||||
| 									? 'font-weight: bold' | ||||
| 									: undefined} | ||||
| 								>${this.unread_status('🔐' + JSON.parse(key).join(','))} | ||||
| 								${(key != '[]' ? JSON.parse(key) : [this.whoami]).map( | ||||
| 									(id) => html` | ||||
| 										<tf-user | ||||
| 											id=${id} | ||||
| 											nolink="true" | ||||
| 											.users=${this.users} | ||||
| 										></tf-user> | ||||
| 									` | ||||
| 								)}</a | ||||
| 							> | ||||
| 						` | ||||
| 					)} | ||||
| 				${Object.keys(this.drafts) | ||||
| 					.sort() | ||||
| 					.map( | ||||
| @@ -380,7 +411,7 @@ class TfTabNewsElement extends LitElement { | ||||
| 		return cache(html` | ||||
| 			${this.render_sidebar()} | ||||
| 			<div | ||||
| 				style="margin-left: 2in; padding: 0px; top: 0; max-height: 100%; overflow: auto; contain: layout" | ||||
| 				style="margin-left: 2in; padding: 0px; top: 0; height: 100vh; max-height: 100%; overflow: auto; contain: layout" | ||||
| 				id="main" | ||||
| 				class="w3-main" | ||||
| 			> | ||||
| @@ -418,6 +449,9 @@ class TfTabNewsElement extends LitElement { | ||||
| 							.drafts=${this.drafts} | ||||
| 							@tf-draft=${this.draft} | ||||
| 							.channel=${this.channel()} | ||||
| 							.recipients=${this.hash.startsWith('#🔐') | ||||
| 								? this.hash.substring('#🔐'.length).split(',') | ||||
| 								: undefined} | ||||
| 						></tf-compose> | ||||
| 					</div> | ||||
| 					${profile} | ||||
| @@ -434,6 +468,7 @@ class TfTabNewsElement extends LitElement { | ||||
| 						.channels_unread=${this.channels_unread} | ||||
| 						.channels_latest=${this.channels_latest} | ||||
| 						.private_messages=${this.private_messages} | ||||
| 						.grouped_private_messages=${this.grouped_private_messages} | ||||
| 						.recent_reactions=${this.recent_reactions} | ||||
| 					></tf-tab-news-feed> | ||||
| 				</div> | ||||
|   | ||||
| @@ -9,6 +9,7 @@ class TfUserElement extends LitElement { | ||||
| 			fallback_name: {type: String}, | ||||
| 			icon_only: {type: Boolean}, | ||||
| 			users: {type: Object}, | ||||
| 			nolink: {type: Boolean}, | ||||
| 		}; | ||||
| 	} | ||||
|  | ||||
| @@ -37,7 +38,9 @@ class TfUserElement extends LitElement { | ||||
| 		let name_string = name ?? this.fallback_name ?? this.id; | ||||
| 		name = this.icon_only | ||||
| 			? undefined | ||||
| 			: html`<a target="_top" href=${'#' + this.id}>${name_string}</a>`; | ||||
| 			: !this.nolink | ||||
| 				? html`<a target="_top" href=${'#' + this.id}>${name_string}</a>` | ||||
| 				: html`<span>${name_string}</span>`; | ||||
|  | ||||
| 		if (user) { | ||||
| 			let image_link = user.image; | ||||
| @@ -56,7 +59,8 @@ class TfUserElement extends LitElement { | ||||
| 			} | ||||
| 		} | ||||
| 		return html` <div | ||||
| 			style="display: inline-block; vertical-align: middle; font-weight: bold; text-wrap: nowrap; max-width: 100%; overflow: hidden; text-overflow: ellipsis" | ||||
| 			style=${'display: inline-block; vertical-align: middle; text-wrap: nowrap; max-width: 100%; overflow: hidden; text-overflow: ellipsis' + | ||||
| 			(this.nolink ? '' : '; font-weight: bold')} | ||||
| 		> | ||||
| 			${image} ${name} | ||||
| 		</div>`; | ||||
|   | ||||
| @@ -104,12 +104,12 @@ export function markdown(md) { | ||||
| 					node.destination.startsWith('@') && | ||||
| 					node.destination.endsWith('.ed25519') | ||||
| 				) { | ||||
| 					node.destination = '#' + node.destination; | ||||
| 					node.destination = '#' + encodeURIComponent(node.destination); | ||||
| 				} else if ( | ||||
| 					node.destination.startsWith('%') && | ||||
| 					node.destination.endsWith('.sha256') | ||||
| 				) { | ||||
| 					node.destination = '#' + node.destination; | ||||
| 					node.destination = '#' + encodeURIComponent(node.destination); | ||||
| 				} else if ( | ||||
| 					node.destination.startsWith('&') && | ||||
| 					node.destination.endsWith('.sha256') | ||||
|   | ||||
| @@ -438,6 +438,9 @@ class TfNavigationElement extends LitElement { | ||||
| 	} | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Create a tf-navigation element. | ||||
|  */ | ||||
| customElements.define('tf-navigation', TfNavigationElement); | ||||
|  | ||||
| /** | ||||
|   | ||||
							
								
								
									
										24
									
								
								core/core.js
									
									
									
									
									
								
							
							
						
						
									
										24
									
								
								core/core.js
									
									
									
									
									
								
							| @@ -199,23 +199,6 @@ async function getProcessBlob(blobId, key, options) { | ||||
| 			let imports = { | ||||
| 				core: { | ||||
| 					broadcast: broadcast.bind(process), | ||||
| 					register: function (eventName, handler) { | ||||
| 						if (!process.eventHandlers[eventName]) { | ||||
| 							process.eventHandlers[eventName] = []; | ||||
| 						} | ||||
| 						process.eventHandlers[eventName].push(handler); | ||||
| 					}, | ||||
| 					unregister: function (eventName, handler) { | ||||
| 						if (process.eventHandlers[eventName]) { | ||||
| 							let index = process.eventHandlers[eventName].indexOf(handler); | ||||
| 							if (index != -1) { | ||||
| 								process.eventHandlers[eventName].splice(index, 1); | ||||
| 							} | ||||
| 							if (process.eventHandlers[eventName].length == 0) { | ||||
| 								delete process.eventHandlers[eventName]; | ||||
| 							} | ||||
| 						} | ||||
| 					}, | ||||
| 					user: getUser(process, process), | ||||
| 					users: async function () { | ||||
| 						try { | ||||
| @@ -703,10 +686,17 @@ async function getProcessBlob(blobId, key, options) { | ||||
| 	return process; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * SSB message added callback. | ||||
|  */ | ||||
| ssb.addEventListener('message', function () { | ||||
| 	broadcastEvent('onMessage', [...arguments]); | ||||
| }); | ||||
|  | ||||
| ssb.addEventListener('blob', function () { | ||||
| 	broadcastEvent('onBlob', [...arguments]); | ||||
| }); | ||||
|  | ||||
| ssb.addEventListener('broadcasts', function () { | ||||
| 	broadcastEvent('onBroadcastsChanged', []); | ||||
| }); | ||||
|   | ||||
| @@ -25,14 +25,14 @@ | ||||
| }: | ||||
| pkgs.stdenv.mkDerivation rec { | ||||
|   pname = "tildefriends"; | ||||
|   version = "0.0.33"; | ||||
|   version = "0.2025.8"; | ||||
|  | ||||
|   src = pkgs.fetchFromGitea { | ||||
|     domain = "dev.tildefriends.net"; | ||||
|     owner = "cory"; | ||||
|     repo = "tildefriends"; | ||||
|     rev = "v${version}"; | ||||
|     hash = "sha256-9D28gmaBTRVyXhY3zZd/W9PsXA1YZt/K69hz41aVP04="; | ||||
|     hash = "sha256-N/5lp8RL19B6Z43kRxx7c01WVJkK44a/wwNgRJPk5uI="; | ||||
|     fetchSubmodules = true; | ||||
|   }; | ||||
|  | ||||
|   | ||||
							
								
								
									
										2
									
								
								deps/codemirror/cm6.js
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								deps/codemirror/cm6.js
									
									
									
									
										vendored
									
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
							
								
								
									
										264
									
								
								deps/codemirror_src/package-lock.json
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						
									
										264
									
								
								deps/codemirror_src/package-lock.json
									
									
									
										generated
									
									
										vendored
									
									
								
							| @@ -19,9 +19,9 @@ | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/@codemirror/autocomplete": { | ||||
|       "version": "6.18.6", | ||||
|       "resolved": "https://registry.npmjs.org/@codemirror/autocomplete/-/autocomplete-6.18.6.tgz", | ||||
|       "integrity": "sha512-PHHBXFomUs5DF+9tCOM/UoW6XQ4R44lLNNhRaW9PKPTU0D7lIjRg3ElxaJnTwsl/oHiR93WSXDBrekhoUGCPtg==", | ||||
|       "version": "6.18.7", | ||||
|       "resolved": "https://registry.npmjs.org/@codemirror/autocomplete/-/autocomplete-6.18.7.tgz", | ||||
|       "integrity": "sha512-8EzdeIoWPJDsMBwz3zdzwXnUpCzMiCyz5/A3FIPpriaclFCGDkAzK13sMcnsu5rowqiyeQN2Vs2TsOcoDPZirQ==", | ||||
|       "license": "MIT", | ||||
|       "dependencies": { | ||||
|         "@codemirror/language": "^6.0.0", | ||||
| @@ -56,9 +56,9 @@ | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/@codemirror/lang-html": { | ||||
|       "version": "6.4.9", | ||||
|       "resolved": "https://registry.npmjs.org/@codemirror/lang-html/-/lang-html-6.4.9.tgz", | ||||
|       "integrity": "sha512-aQv37pIMSlueybId/2PVSP6NPnmurFDVmZwzc7jszd2KAF8qd4VBbvNYPXWQq90WIARjsdVkPbw29pszmHws3Q==", | ||||
|       "version": "6.4.10", | ||||
|       "resolved": "https://registry.npmjs.org/@codemirror/lang-html/-/lang-html-6.4.10.tgz", | ||||
|       "integrity": "sha512-h/SceTVsN5r+WE+TVP2g3KDvNoSzbSrtZXCKo4vkKdbfT5t4otuVgngGdFukOO/rwRD2++pCxoh6xD4TEVMkQA==", | ||||
|       "license": "MIT", | ||||
|       "dependencies": { | ||||
|         "@codemirror/autocomplete": "^6.0.0", | ||||
| @@ -98,9 +98,9 @@ | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/@codemirror/language": { | ||||
|       "version": "6.11.2", | ||||
|       "resolved": "https://registry.npmjs.org/@codemirror/language/-/language-6.11.2.tgz", | ||||
|       "integrity": "sha512-p44TsNArL4IVXDTbapUmEkAlvWs2CFQbcfc0ymDsis1kH2wh0gcY96AS29c/vp2d0y2Tquk1EDSaawpzilUiAw==", | ||||
|       "version": "6.11.3", | ||||
|       "resolved": "https://registry.npmjs.org/@codemirror/language/-/language-6.11.3.tgz", | ||||
|       "integrity": "sha512-9HBM2XnwDj7fnu0551HkGdrUrrqmYq/WC5iv6nbY2WdicXdGbhR/gfbZOH73Aqj4351alY1+aoG9rCNfiwS1RA==", | ||||
|       "license": "MIT", | ||||
|       "dependencies": { | ||||
|         "@codemirror/state": "^6.0.0", | ||||
| @@ -155,9 +155,9 @@ | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/@codemirror/view": { | ||||
|       "version": "6.38.1", | ||||
|       "resolved": "https://registry.npmjs.org/@codemirror/view/-/view-6.38.1.tgz", | ||||
|       "integrity": "sha512-RmTOkE7hRU3OVREqFVITWHz6ocgBjv08GoePscAakgVQfciA3SGCEk7mb9IzwW61cKKmlTpHXG6DUE5Ubx+MGQ==", | ||||
|       "version": "6.38.3", | ||||
|       "resolved": "https://registry.npmjs.org/@codemirror/view/-/view-6.38.3.tgz", | ||||
|       "integrity": "sha512-x2t87+oqwB1mduiQZ6huIghjMt4uZKFEdj66IcXw7+a5iBEvv9lh7EWDRHI7crnD4BMGpnyq/RzmCGbiEZLcvQ==", | ||||
|       "license": "MIT", | ||||
|       "dependencies": { | ||||
|         "@codemirror/state": "^6.5.0", | ||||
| @@ -167,9 +167,9 @@ | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/@jridgewell/gen-mapping": { | ||||
|       "version": "0.3.12", | ||||
|       "resolved": "https://registry.npmjs.org/@jridgewell/gen-mapping/-/gen-mapping-0.3.12.tgz", | ||||
|       "integrity": "sha512-OuLGC46TjB5BbN1dH8JULVVZY4WTdkF7tV9Ys6wLL1rubZnCMstOhNHueU5bLCrnRuDhKPDM4g6sw4Bel5Gzqg==", | ||||
|       "version": "0.3.13", | ||||
|       "resolved": "https://registry.npmjs.org/@jridgewell/gen-mapping/-/gen-mapping-0.3.13.tgz", | ||||
|       "integrity": "sha512-2kkt/7niJ6MgEPxF0bYdQ6etZaA+fQvDcLKckhy1yIQOzaoKjBBjSj63/aLVjYE3qhRt5dvM+uUyfCg6UKCBbA==", | ||||
|       "dev": true, | ||||
|       "license": "MIT", | ||||
|       "dependencies": { | ||||
| @@ -188,9 +188,9 @@ | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/@jridgewell/source-map": { | ||||
|       "version": "0.3.10", | ||||
|       "resolved": "https://registry.npmjs.org/@jridgewell/source-map/-/source-map-0.3.10.tgz", | ||||
|       "integrity": "sha512-0pPkgz9dY+bijgistcTTJ5mR+ocqRXLuhXHYdzoMmmoJ2C9S46RCm2GMUbatPEUK9Yjy26IrAy8D/M00lLkv+Q==", | ||||
|       "version": "0.3.11", | ||||
|       "resolved": "https://registry.npmjs.org/@jridgewell/source-map/-/source-map-0.3.11.tgz", | ||||
|       "integrity": "sha512-ZMp1V8ZFcPG5dIWnQLr3NSI1MiCU7UETdS/A0G8V/XWHvJv3ZsFqutJn1Y5RPmAPX6F3BiE397OqveU/9NCuIA==", | ||||
|       "dev": true, | ||||
|       "license": "MIT", | ||||
|       "dependencies": { | ||||
| @@ -199,16 +199,16 @@ | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/@jridgewell/sourcemap-codec": { | ||||
|       "version": "1.5.4", | ||||
|       "resolved": "https://registry.npmjs.org/@jridgewell/sourcemap-codec/-/sourcemap-codec-1.5.4.tgz", | ||||
|       "integrity": "sha512-VT2+G1VQs/9oz078bLrYbecdZKs912zQlkelYpuf+SXF+QvZDYJlbx/LSx+meSAwdDFnF8FVXW92AVjjkVmgFw==", | ||||
|       "version": "1.5.5", | ||||
|       "resolved": "https://registry.npmjs.org/@jridgewell/sourcemap-codec/-/sourcemap-codec-1.5.5.tgz", | ||||
|       "integrity": "sha512-cYQ9310grqxueWbl+WuIUIaiUaDcj7WOq5fVhEljNVgRfOUhY9fy2zTvfoqWsnebh8Sl70VScFbICvJnLKB0Og==", | ||||
|       "dev": true, | ||||
|       "license": "MIT" | ||||
|     }, | ||||
|     "node_modules/@jridgewell/trace-mapping": { | ||||
|       "version": "0.3.29", | ||||
|       "resolved": "https://registry.npmjs.org/@jridgewell/trace-mapping/-/trace-mapping-0.3.29.tgz", | ||||
|       "integrity": "sha512-uw6guiW/gcAGPDhLmd77/6lW8QLeiV5RUTsAX46Db6oLhGaVj4lhnPwb184s1bkc8kdVg/+h988dro8GRDpmYQ==", | ||||
|       "version": "0.3.31", | ||||
|       "resolved": "https://registry.npmjs.org/@jridgewell/trace-mapping/-/trace-mapping-0.3.31.tgz", | ||||
|       "integrity": "sha512-zzNR+SdQSDJzc8joaeP8QQoCQr8NuYx2dIIytl1QeBEZHJ9uW6hebsrYgbz8hJwUQao3TWCMtmfV8Nu1twOLAw==", | ||||
|       "dev": true, | ||||
|       "license": "MIT", | ||||
|       "dependencies": { | ||||
| @@ -254,9 +254,9 @@ | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/@lezer/javascript": { | ||||
|       "version": "1.5.1", | ||||
|       "resolved": "https://registry.npmjs.org/@lezer/javascript/-/javascript-1.5.1.tgz", | ||||
|       "integrity": "sha512-ATOImjeVJuvgm3JQ/bpo2Tmv55HSScE2MTPnKRMRIPx2cLhHGyX2VnqpHhtIV1tVzIjZDbcWQm+NCTF40ggZVw==", | ||||
|       "version": "1.5.4", | ||||
|       "resolved": "https://registry.npmjs.org/@lezer/javascript/-/javascript-1.5.4.tgz", | ||||
|       "integrity": "sha512-vvYx3MhWqeZtGPwDStM2dwgljd5smolYD2lR2UyFcHfxbBQebqx8yjmFmxtJ/E6nN6u1D9srOiVWm3Rb4tmcUA==", | ||||
|       "license": "MIT", | ||||
|       "dependencies": { | ||||
|         "@lezer/common": "^1.2.0", | ||||
| @@ -338,9 +338,9 @@ | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/@rollup/pluginutils": { | ||||
|       "version": "5.2.0", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/pluginutils/-/pluginutils-5.2.0.tgz", | ||||
|       "integrity": "sha512-qWJ2ZTbmumwiLFomfzTyt5Kng4hwPi9rwCYN4SHb6eaRU1KNO4ccxINHr/VhH4GgPlt1XfSTLX2LBTme8ne4Zw==", | ||||
|       "version": "5.3.0", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/pluginutils/-/pluginutils-5.3.0.tgz", | ||||
|       "integrity": "sha512-5EdhGZtnu3V88ces7s53hhfK5KSASnJZv8Lulpc04cWO3REESroJXg73DFsOmgbU2BhwV0E20bu2IDZb3VKW4Q==", | ||||
|       "license": "MIT", | ||||
|       "dependencies": { | ||||
|         "@types/estree": "^1.0.0", | ||||
| @@ -360,9 +360,9 @@ | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/@rollup/rollup-android-arm-eabi": { | ||||
|       "version": "4.46.0", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-android-arm-eabi/-/rollup-android-arm-eabi-4.46.0.tgz", | ||||
|       "integrity": "sha512-9f3nSTFI2ivfxc7/tHBHcJ8pRnp8ROrELvsVprlQPVvcZ+j5zztYd+PTJGpyIOAdTvNwNrpCXswKSeoQcyGjMQ==", | ||||
|       "version": "4.52.2", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-android-arm-eabi/-/rollup-android-arm-eabi-4.52.2.tgz", | ||||
|       "integrity": "sha512-o3pcKzJgSGt4d74lSZ+OCnHwkKBeAbFDmbEm5gg70eA8VkyCuC/zV9TwBnmw6VjDlRdF4Pshfb+WE9E6XY1PoQ==", | ||||
|       "cpu": [ | ||||
|         "arm" | ||||
|       ], | ||||
| @@ -373,9 +373,9 @@ | ||||
|       ] | ||||
|     }, | ||||
|     "node_modules/@rollup/rollup-android-arm64": { | ||||
|       "version": "4.46.0", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-android-arm64/-/rollup-android-arm64-4.46.0.tgz", | ||||
|       "integrity": "sha512-tFZSEhqJ8Yrpe50TzOdeoYi72gi/jsnT7y8Qrozf3cNu28WX+s6I3XzEPUAqoaT9SAS8Xz9AzGTFlxxCH/w20w==", | ||||
|       "version": "4.52.2", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-android-arm64/-/rollup-android-arm64-4.52.2.tgz", | ||||
|       "integrity": "sha512-cqFSWO5tX2vhC9hJTK8WAiPIm4Q8q/cU8j2HQA0L3E1uXvBYbOZMhE2oFL8n2pKB5sOCHY6bBuHaRwG7TkfJyw==", | ||||
|       "cpu": [ | ||||
|         "arm64" | ||||
|       ], | ||||
| @@ -386,9 +386,9 @@ | ||||
|       ] | ||||
|     }, | ||||
|     "node_modules/@rollup/rollup-darwin-arm64": { | ||||
|       "version": "4.46.0", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-darwin-arm64/-/rollup-darwin-arm64-4.46.0.tgz", | ||||
|       "integrity": "sha512-+DikIIs+p6yU2hF51UaWG8BnHbq90X0QIOt5zqSKSZxY+G3qqdLih214e9InJal21af2PuuxkDectetGfbVPJw==", | ||||
|       "version": "4.52.2", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-darwin-arm64/-/rollup-darwin-arm64-4.52.2.tgz", | ||||
|       "integrity": "sha512-vngduywkkv8Fkh3wIZf5nFPXzWsNsVu1kvtLETWxTFf/5opZmflgVSeLgdHR56RQh71xhPhWoOkEBvbehwTlVA==", | ||||
|       "cpu": [ | ||||
|         "arm64" | ||||
|       ], | ||||
| @@ -399,9 +399,9 @@ | ||||
|       ] | ||||
|     }, | ||||
|     "node_modules/@rollup/rollup-darwin-x64": { | ||||
|       "version": "4.46.0", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-darwin-x64/-/rollup-darwin-x64-4.46.0.tgz", | ||||
|       "integrity": "sha512-5a+NofhdEB/WimSlFMskbFQn1vqz1FWryYpA99trmZGO6qEmiS0IsX6w4B3d91U878Q2ZQdiaFF1gxX4P147og==", | ||||
|       "version": "4.52.2", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-darwin-x64/-/rollup-darwin-x64-4.52.2.tgz", | ||||
|       "integrity": "sha512-h11KikYrUCYTrDj6h939hhMNlqU2fo/X4NB0OZcys3fya49o1hmFaczAiJWVAFgrM1NCP6RrO7lQKeVYSKBPSQ==", | ||||
|       "cpu": [ | ||||
|         "x64" | ||||
|       ], | ||||
| @@ -412,9 +412,9 @@ | ||||
|       ] | ||||
|     }, | ||||
|     "node_modules/@rollup/rollup-freebsd-arm64": { | ||||
|       "version": "4.46.0", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-freebsd-arm64/-/rollup-freebsd-arm64-4.46.0.tgz", | ||||
|       "integrity": "sha512-igr/RlKPS3OCy4jD3XBmAmo3UAcNZkJSubRsw1JeM8bAbwf15k/3eMZXD91bnjheijJiOJcga3kfCLKjV8IXNg==", | ||||
|       "version": "4.52.2", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-freebsd-arm64/-/rollup-freebsd-arm64-4.52.2.tgz", | ||||
|       "integrity": "sha512-/eg4CI61ZUkLXxMHyVlmlGrSQZ34xqWlZNW43IAU4RmdzWEx0mQJ2mN/Cx4IHLVZFL6UBGAh+/GXhgvGb+nVxw==", | ||||
|       "cpu": [ | ||||
|         "arm64" | ||||
|       ], | ||||
| @@ -425,9 +425,9 @@ | ||||
|       ] | ||||
|     }, | ||||
|     "node_modules/@rollup/rollup-freebsd-x64": { | ||||
|       "version": "4.46.0", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-freebsd-x64/-/rollup-freebsd-x64-4.46.0.tgz", | ||||
|       "integrity": "sha512-MdigWzPSHlQzB1xZ+MdFDWTAH+kcn7UxjEBoOKuaso7z1DRlnAnrknB1mTtNOQ+GdPI8xgExAGwHeqQjntR0Cg==", | ||||
|       "version": "4.52.2", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-freebsd-x64/-/rollup-freebsd-x64-4.52.2.tgz", | ||||
|       "integrity": "sha512-QOWgFH5X9+p+S1NAfOqc0z8qEpJIoUHf7OWjNUGOeW18Mx22lAUOiA9b6r2/vpzLdfxi/f+VWsYjUOMCcYh0Ng==", | ||||
|       "cpu": [ | ||||
|         "x64" | ||||
|       ], | ||||
| @@ -438,9 +438,9 @@ | ||||
|       ] | ||||
|     }, | ||||
|     "node_modules/@rollup/rollup-linux-arm-gnueabihf": { | ||||
|       "version": "4.46.0", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm-gnueabihf/-/rollup-linux-arm-gnueabihf-4.46.0.tgz", | ||||
|       "integrity": "sha512-dmZseE0ZwA/4yy1+BwFrDqFTjjNg24GO9xSrb1weVbt6AFkhp5pz1gVS7IMtfIvoWy8yp6q/zN0bKnefRUImvQ==", | ||||
|       "version": "4.52.2", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm-gnueabihf/-/rollup-linux-arm-gnueabihf-4.52.2.tgz", | ||||
|       "integrity": "sha512-kDWSPafToDd8LcBYd1t5jw7bD5Ojcu12S3uT372e5HKPzQt532vW+rGFFOaiR0opxePyUkHrwz8iWYEyH1IIQA==", | ||||
|       "cpu": [ | ||||
|         "arm" | ||||
|       ], | ||||
| @@ -451,9 +451,9 @@ | ||||
|       ] | ||||
|     }, | ||||
|     "node_modules/@rollup/rollup-linux-arm-musleabihf": { | ||||
|       "version": "4.46.0", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm-musleabihf/-/rollup-linux-arm-musleabihf-4.46.0.tgz", | ||||
|       "integrity": "sha512-fzhfn6p9Cfm3W8UrWKIa4l7Wfjs/KGdgaswMBBE3KY3Ta43jg2XsPrAtfezHpsRk0Nx+TFuS3hZk/To2N5kFPQ==", | ||||
|       "version": "4.52.2", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm-musleabihf/-/rollup-linux-arm-musleabihf-4.52.2.tgz", | ||||
|       "integrity": "sha512-gKm7Mk9wCv6/rkzwCiUC4KnevYhlf8ztBrDRT9g/u//1fZLapSRc+eDZj2Eu2wpJ+0RzUKgtNijnVIB4ZxyL+w==", | ||||
|       "cpu": [ | ||||
|         "arm" | ||||
|       ], | ||||
| @@ -464,9 +464,9 @@ | ||||
|       ] | ||||
|     }, | ||||
|     "node_modules/@rollup/rollup-linux-arm64-gnu": { | ||||
|       "version": "4.46.0", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm64-gnu/-/rollup-linux-arm64-gnu-4.46.0.tgz", | ||||
|       "integrity": "sha512-vVDD+iPDPmJQ5nAQ5Tifq3ywdv60FartglFI8VOCK+hcU9aoG0qlQTsDJP97O5yiTaTqlneZWoARMcVC5nyUoQ==", | ||||
|       "version": "4.52.2", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm64-gnu/-/rollup-linux-arm64-gnu-4.52.2.tgz", | ||||
|       "integrity": "sha512-66lA8vnj5mB/rtDNwPgrrKUOtCLVQypkyDa2gMfOefXK6rcZAxKLO9Fy3GkW8VkPnENv9hBkNOFfGLf6rNKGUg==", | ||||
|       "cpu": [ | ||||
|         "arm64" | ||||
|       ], | ||||
| @@ -477,9 +477,9 @@ | ||||
|       ] | ||||
|     }, | ||||
|     "node_modules/@rollup/rollup-linux-arm64-musl": { | ||||
|       "version": "4.46.0", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm64-musl/-/rollup-linux-arm64-musl-4.46.0.tgz", | ||||
|       "integrity": "sha512-0d0jx08fzDHCzXqrtCMEEyxKU0SvJrWmUjUDE2/KDQ2UDJql0tfiwYvEx1oHELClKO8CNdE+AGJj+RqXscZpdQ==", | ||||
|       "version": "4.52.2", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm64-musl/-/rollup-linux-arm64-musl-4.52.2.tgz", | ||||
|       "integrity": "sha512-s+OPucLNdJHvuZHuIz2WwncJ+SfWHFEmlC5nKMUgAelUeBUnlB4wt7rXWiyG4Zn07uY2Dd+SGyVa9oyLkVGOjA==", | ||||
|       "cpu": [ | ||||
|         "arm64" | ||||
|       ], | ||||
| @@ -489,10 +489,10 @@ | ||||
|         "linux" | ||||
|       ] | ||||
|     }, | ||||
|     "node_modules/@rollup/rollup-linux-loongarch64-gnu": { | ||||
|       "version": "4.46.0", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-loongarch64-gnu/-/rollup-linux-loongarch64-gnu-4.46.0.tgz", | ||||
|       "integrity": "sha512-XBYu9oW9eKJadWn8M7hkTZsD4yG+RrsTrVEgyKwb4L72cpJjRbRboTG9Lg9fec8MxJp/cfTHAocg4mnismQR8A==", | ||||
|     "node_modules/@rollup/rollup-linux-loong64-gnu": { | ||||
|       "version": "4.52.2", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-loong64-gnu/-/rollup-linux-loong64-gnu-4.52.2.tgz", | ||||
|       "integrity": "sha512-8wTRM3+gVMDLLDdaT6tKmOE3lJyRy9NpJUS/ZRWmLCmOPIJhVyXwjBo+XbrrwtV33Em1/eCTd5TuGJm4+DmYjw==", | ||||
|       "cpu": [ | ||||
|         "loong64" | ||||
|       ], | ||||
| @@ -503,9 +503,9 @@ | ||||
|       ] | ||||
|     }, | ||||
|     "node_modules/@rollup/rollup-linux-ppc64-gnu": { | ||||
|       "version": "4.46.0", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-ppc64-gnu/-/rollup-linux-ppc64-gnu-4.46.0.tgz", | ||||
|       "integrity": "sha512-wJaRvcT17PoOK6Ggcfo3nouFlybHvARBS4jzT0PC/lg17fIJHcDS2fZz3sD+iA4nRlho2zE6OGbU0HvwATdokQ==", | ||||
|       "version": "4.52.2", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-ppc64-gnu/-/rollup-linux-ppc64-gnu-4.52.2.tgz", | ||||
|       "integrity": "sha512-6yqEfgJ1anIeuP2P/zhtfBlDpXUb80t8DpbYwXQ3bQd95JMvUaqiX+fKqYqUwZXqdJDd8xdilNtsHM2N0cFm6A==", | ||||
|       "cpu": [ | ||||
|         "ppc64" | ||||
|       ], | ||||
| @@ -516,9 +516,9 @@ | ||||
|       ] | ||||
|     }, | ||||
|     "node_modules/@rollup/rollup-linux-riscv64-gnu": { | ||||
|       "version": "4.46.0", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-riscv64-gnu/-/rollup-linux-riscv64-gnu-4.46.0.tgz", | ||||
|       "integrity": "sha512-GZ5bkMFteAGkcmh8x0Ok4LSa+L62Ez0tMsHPX6JtR0wl4Xc3bQcrFHDiR5DGLEDFtGrXih4Nd/UDaFqs968/wA==", | ||||
|       "version": "4.52.2", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-riscv64-gnu/-/rollup-linux-riscv64-gnu-4.52.2.tgz", | ||||
|       "integrity": "sha512-sshYUiYVSEI2B6dp4jMncwxbrUqRdNApF2c3bhtLAU0qA8Lrri0p0NauOsTWh3yCCCDyBOjESHMExonp7Nzc0w==", | ||||
|       "cpu": [ | ||||
|         "riscv64" | ||||
|       ], | ||||
| @@ -529,9 +529,9 @@ | ||||
|       ] | ||||
|     }, | ||||
|     "node_modules/@rollup/rollup-linux-riscv64-musl": { | ||||
|       "version": "4.46.0", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-riscv64-musl/-/rollup-linux-riscv64-musl-4.46.0.tgz", | ||||
|       "integrity": "sha512-7CjPw6FflFsVOUfWOrVrREiV3IYXG4RzZ1ZQUaT3BtSK8YXN6x286o+sruPZJESIaPebYuFowmg54ZdrkVBYog==", | ||||
|       "version": "4.52.2", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-riscv64-musl/-/rollup-linux-riscv64-musl-4.52.2.tgz", | ||||
|       "integrity": "sha512-duBLgd+3pqC4MMwBrKkFxaZerUxZcYApQVC5SdbF5/e/589GwVvlRUnyqMFbM8iUSb1BaoX/3fRL7hB9m2Pj8Q==", | ||||
|       "cpu": [ | ||||
|         "riscv64" | ||||
|       ], | ||||
| @@ -542,9 +542,9 @@ | ||||
|       ] | ||||
|     }, | ||||
|     "node_modules/@rollup/rollup-linux-s390x-gnu": { | ||||
|       "version": "4.46.0", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-s390x-gnu/-/rollup-linux-s390x-gnu-4.46.0.tgz", | ||||
|       "integrity": "sha512-nmvnl0ZiuysltcB/cKjUh40Rx4FbSyueERDsl2FLvLYr6pCgSsvGr3SocUT84svSpmloS7f1DRWqtRha74Gi1w==", | ||||
|       "version": "4.52.2", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-s390x-gnu/-/rollup-linux-s390x-gnu-4.52.2.tgz", | ||||
|       "integrity": "sha512-tzhYJJidDUVGMgVyE+PmxENPHlvvqm1KILjjZhB8/xHYqAGeizh3GBGf9u6WdJpZrz1aCpIIHG0LgJgH9rVjHQ==", | ||||
|       "cpu": [ | ||||
|         "s390x" | ||||
|       ], | ||||
| @@ -555,9 +555,9 @@ | ||||
|       ] | ||||
|     }, | ||||
|     "node_modules/@rollup/rollup-linux-x64-gnu": { | ||||
|       "version": "4.46.0", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-x64-gnu/-/rollup-linux-x64-gnu-4.46.0.tgz", | ||||
|       "integrity": "sha512-Cv+moII5C8RM6gZbR3cb21o6rquVDZrN2o81maROg1LFzBz2dZUwIQSxFA8GtGZ/F2KtsqQ2z3eFPBb6akvQNg==", | ||||
|       "version": "4.52.2", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-x64-gnu/-/rollup-linux-x64-gnu-4.52.2.tgz", | ||||
|       "integrity": "sha512-opH8GSUuVcCSSyHHcl5hELrmnk4waZoVpgn/4FDao9iyE4WpQhyWJ5ryl5M3ocp4qkRuHfyXnGqg8M9oKCEKRA==", | ||||
|       "cpu": [ | ||||
|         "x64" | ||||
|       ], | ||||
| @@ -568,9 +568,9 @@ | ||||
|       ] | ||||
|     }, | ||||
|     "node_modules/@rollup/rollup-linux-x64-musl": { | ||||
|       "version": "4.46.0", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-x64-musl/-/rollup-linux-x64-musl-4.46.0.tgz", | ||||
|       "integrity": "sha512-PHcMG8DZTM9RCIjp8QIfN0VYtX0TtBPnWOTRurFhoCDoi9zptUZL2k7pCs+5rgut7JAiUsYy+huyhVKPcmxoog==", | ||||
|       "version": "4.52.2", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-x64-musl/-/rollup-linux-x64-musl-4.52.2.tgz", | ||||
|       "integrity": "sha512-LSeBHnGli1pPKVJ79ZVJgeZWWZXkEe/5o8kcn23M8eMKCUANejchJbF/JqzM4RRjOJfNRhKJk8FuqL1GKjF5oQ==", | ||||
|       "cpu": [ | ||||
|         "x64" | ||||
|       ], | ||||
| @@ -580,10 +580,23 @@ | ||||
|         "linux" | ||||
|       ] | ||||
|     }, | ||||
|     "node_modules/@rollup/rollup-openharmony-arm64": { | ||||
|       "version": "4.52.2", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-openharmony-arm64/-/rollup-openharmony-arm64-4.52.2.tgz", | ||||
|       "integrity": "sha512-uPj7MQ6/s+/GOpolavm6BPo+6CbhbKYyZHUDvZ/SmJM7pfDBgdGisFX3bY/CBDMg2ZO4utfhlApkSfZ92yXw7Q==", | ||||
|       "cpu": [ | ||||
|         "arm64" | ||||
|       ], | ||||
|       "license": "MIT", | ||||
|       "optional": true, | ||||
|       "os": [ | ||||
|         "openharmony" | ||||
|       ] | ||||
|     }, | ||||
|     "node_modules/@rollup/rollup-win32-arm64-msvc": { | ||||
|       "version": "4.46.0", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-arm64-msvc/-/rollup-win32-arm64-msvc-4.46.0.tgz", | ||||
|       "integrity": "sha512-1SI/Rd47e8aQJeFWMDg16ET+fjvCcD/CzeaRmIEPmb05hx+3cCcwIF4ebUag4yTt/D1peE+Mgp0+Po3M358cAA==", | ||||
|       "version": "4.52.2", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-arm64-msvc/-/rollup-win32-arm64-msvc-4.52.2.tgz", | ||||
|       "integrity": "sha512-Z9MUCrSgIaUeeHAiNkm3cQyst2UhzjPraR3gYYfOjAuZI7tcFRTOD+4cHLPoS/3qinchth+V56vtqz1Tv+6KPA==", | ||||
|       "cpu": [ | ||||
|         "arm64" | ||||
|       ], | ||||
| @@ -594,9 +607,9 @@ | ||||
|       ] | ||||
|     }, | ||||
|     "node_modules/@rollup/rollup-win32-ia32-msvc": { | ||||
|       "version": "4.46.0", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-ia32-msvc/-/rollup-win32-ia32-msvc-4.46.0.tgz", | ||||
|       "integrity": "sha512-JwOCYxmumFDfDhx4kNyz6kTVK3gWzBIvVdMNzQMRDubcoGRDniOOmo6DDNP42qwZx3Bp9/6vWJ+kNzNqXoHmeA==", | ||||
|       "version": "4.52.2", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-ia32-msvc/-/rollup-win32-ia32-msvc-4.52.2.tgz", | ||||
|       "integrity": "sha512-+GnYBmpjldD3XQd+HMejo+0gJGwYIOfFeoBQv32xF/RUIvccUz20/V6Otdv+57NE70D5pa8W/jVGDoGq0oON4A==", | ||||
|       "cpu": [ | ||||
|         "ia32" | ||||
|       ], | ||||
| @@ -606,10 +619,23 @@ | ||||
|         "win32" | ||||
|       ] | ||||
|     }, | ||||
|     "node_modules/@rollup/rollup-win32-x64-gnu": { | ||||
|       "version": "4.52.2", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-x64-gnu/-/rollup-win32-x64-gnu-4.52.2.tgz", | ||||
|       "integrity": "sha512-ApXFKluSB6kDQkAqZOKXBjiaqdF1BlKi+/eqnYe9Ee7U2K3pUDKsIyr8EYm/QDHTJIM+4X+lI0gJc3TTRhd+dA==", | ||||
|       "cpu": [ | ||||
|         "x64" | ||||
|       ], | ||||
|       "license": "MIT", | ||||
|       "optional": true, | ||||
|       "os": [ | ||||
|         "win32" | ||||
|       ] | ||||
|     }, | ||||
|     "node_modules/@rollup/rollup-win32-x64-msvc": { | ||||
|       "version": "4.46.0", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-x64-msvc/-/rollup-win32-x64-msvc-4.46.0.tgz", | ||||
|       "integrity": "sha512-IPMIfrfkG1GaEXi+JSsQEx8x9b4b+hRZXO7KYc2pKio3zO2/VDXDs6B9Ts/nnO+25Fk1tdAVtUn60HKKPPzDig==", | ||||
|       "version": "4.52.2", | ||||
|       "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-x64-msvc/-/rollup-win32-x64-msvc-4.52.2.tgz", | ||||
|       "integrity": "sha512-ARz+Bs8kY6FtitYM96PqPEVvPXqEZmPZsSkXvyX19YzDqkCaIlhCieLLMI5hxO9SRZ2XtCtm8wxhy0iJ2jxNfw==", | ||||
|       "cpu": [ | ||||
|         "x64" | ||||
|       ], | ||||
| @@ -799,9 +825,9 @@ | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/rollup": { | ||||
|       "version": "4.46.0", | ||||
|       "resolved": "https://registry.npmjs.org/rollup/-/rollup-4.46.0.tgz", | ||||
|       "integrity": "sha512-ONmkT3Ud3IfW15nl7l4qAZko5/2iZ5ALVBDh02ZSZ5IGVLJSYkRcRa3iB58VyEIyoofs9m2xdVrm+lTi97+3pw==", | ||||
|       "version": "4.52.2", | ||||
|       "resolved": "https://registry.npmjs.org/rollup/-/rollup-4.52.2.tgz", | ||||
|       "integrity": "sha512-I25/2QgoROE1vYV+NQ1En9T9UFB9Cmfm2CJ83zZOlaDpvz29wGQSZXWKw7MiNXau7wYgB/T9fVIdIuEQ+KbiiA==", | ||||
|       "license": "MIT", | ||||
|       "dependencies": { | ||||
|         "@types/estree": "1.0.8" | ||||
| @@ -814,26 +840,28 @@ | ||||
|         "npm": ">=8.0.0" | ||||
|       }, | ||||
|       "optionalDependencies": { | ||||
|         "@rollup/rollup-android-arm-eabi": "4.46.0", | ||||
|         "@rollup/rollup-android-arm64": "4.46.0", | ||||
|         "@rollup/rollup-darwin-arm64": "4.46.0", | ||||
|         "@rollup/rollup-darwin-x64": "4.46.0", | ||||
|         "@rollup/rollup-freebsd-arm64": "4.46.0", | ||||
|         "@rollup/rollup-freebsd-x64": "4.46.0", | ||||
|         "@rollup/rollup-linux-arm-gnueabihf": "4.46.0", | ||||
|         "@rollup/rollup-linux-arm-musleabihf": "4.46.0", | ||||
|         "@rollup/rollup-linux-arm64-gnu": "4.46.0", | ||||
|         "@rollup/rollup-linux-arm64-musl": "4.46.0", | ||||
|         "@rollup/rollup-linux-loongarch64-gnu": "4.46.0", | ||||
|         "@rollup/rollup-linux-ppc64-gnu": "4.46.0", | ||||
|         "@rollup/rollup-linux-riscv64-gnu": "4.46.0", | ||||
|         "@rollup/rollup-linux-riscv64-musl": "4.46.0", | ||||
|         "@rollup/rollup-linux-s390x-gnu": "4.46.0", | ||||
|         "@rollup/rollup-linux-x64-gnu": "4.46.0", | ||||
|         "@rollup/rollup-linux-x64-musl": "4.46.0", | ||||
|         "@rollup/rollup-win32-arm64-msvc": "4.46.0", | ||||
|         "@rollup/rollup-win32-ia32-msvc": "4.46.0", | ||||
|         "@rollup/rollup-win32-x64-msvc": "4.46.0", | ||||
|         "@rollup/rollup-android-arm-eabi": "4.52.2", | ||||
|         "@rollup/rollup-android-arm64": "4.52.2", | ||||
|         "@rollup/rollup-darwin-arm64": "4.52.2", | ||||
|         "@rollup/rollup-darwin-x64": "4.52.2", | ||||
|         "@rollup/rollup-freebsd-arm64": "4.52.2", | ||||
|         "@rollup/rollup-freebsd-x64": "4.52.2", | ||||
|         "@rollup/rollup-linux-arm-gnueabihf": "4.52.2", | ||||
|         "@rollup/rollup-linux-arm-musleabihf": "4.52.2", | ||||
|         "@rollup/rollup-linux-arm64-gnu": "4.52.2", | ||||
|         "@rollup/rollup-linux-arm64-musl": "4.52.2", | ||||
|         "@rollup/rollup-linux-loong64-gnu": "4.52.2", | ||||
|         "@rollup/rollup-linux-ppc64-gnu": "4.52.2", | ||||
|         "@rollup/rollup-linux-riscv64-gnu": "4.52.2", | ||||
|         "@rollup/rollup-linux-riscv64-musl": "4.52.2", | ||||
|         "@rollup/rollup-linux-s390x-gnu": "4.52.2", | ||||
|         "@rollup/rollup-linux-x64-gnu": "4.52.2", | ||||
|         "@rollup/rollup-linux-x64-musl": "4.52.2", | ||||
|         "@rollup/rollup-openharmony-arm64": "4.52.2", | ||||
|         "@rollup/rollup-win32-arm64-msvc": "4.52.2", | ||||
|         "@rollup/rollup-win32-ia32-msvc": "4.52.2", | ||||
|         "@rollup/rollup-win32-x64-gnu": "4.52.2", | ||||
|         "@rollup/rollup-win32-x64-msvc": "4.52.2", | ||||
|         "fsevents": "~2.3.2" | ||||
|       } | ||||
|     }, | ||||
| @@ -915,14 +943,14 @@ | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/terser": { | ||||
|       "version": "5.43.1", | ||||
|       "resolved": "https://registry.npmjs.org/terser/-/terser-5.43.1.tgz", | ||||
|       "integrity": "sha512-+6erLbBm0+LROX2sPXlUYx/ux5PyE9K/a92Wrt6oA+WDAoFTdpHE5tCYCI5PNzq2y8df4rA+QgHLJuR4jNymsg==", | ||||
|       "version": "5.44.0", | ||||
|       "resolved": "https://registry.npmjs.org/terser/-/terser-5.44.0.tgz", | ||||
|       "integrity": "sha512-nIVck8DK+GM/0Frwd+nIhZ84pR/BX7rmXMfYwyg+Sri5oGVE99/E3KvXqpC2xHFxyqXyGHTKBSioxxplrO4I4w==", | ||||
|       "dev": true, | ||||
|       "license": "BSD-2-Clause", | ||||
|       "dependencies": { | ||||
|         "@jridgewell/source-map": "^0.3.3", | ||||
|         "acorn": "^8.14.0", | ||||
|         "acorn": "^8.15.0", | ||||
|         "commander": "^2.20.0", | ||||
|         "source-map-support": "~0.5.20" | ||||
|       }, | ||||
|   | ||||
							
								
								
									
										2
									
								
								deps/openssl_src
									
									
									
									
										vendored
									
									
								
							
							
								
								
								
								
								
							
						
						
									
										2
									
								
								deps/openssl_src
									
									
									
									
										vendored
									
									
								
							 Submodule deps/openssl_src updated: 0893a62353...c4da9ac23d
									
								
							
							
								
								
									
										2
									
								
								deps/quickjs
									
									
									
									
										vendored
									
									
								
							
							
								
								
								
								
								
							
						
						
									
										2
									
								
								deps/quickjs
									
									
									
									
										vendored
									
									
								
							 Submodule deps/quickjs updated: 19abf1888d...fa628f8c52
									
								
							
							
								
								
									
										6
									
								
								flake.lock
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										6
									
								
								flake.lock
									
									
									
										generated
									
									
									
								
							| @@ -20,11 +20,11 @@ | ||||
|     }, | ||||
|     "nixpkgs": { | ||||
|       "locked": { | ||||
|         "lastModified": 1753749649, | ||||
|         "narHash": "sha256-+jkEZxs7bfOKfBIk430K+tK9IvXlwzqQQnppC2ZKFj4=", | ||||
|         "lastModified": 1756217674, | ||||
|         "narHash": "sha256-TH1SfSP523QI7kcPiNtMAEuwZR3Jdz0MCDXPs7TS8uo=", | ||||
|         "owner": "NixOS", | ||||
|         "repo": "nixpkgs", | ||||
|         "rev": "1f08a4df998e21f4e8be8fb6fbf61d11a1a5076a", | ||||
|         "rev": "4e7667a90c167f7a81d906e5a75cba4ad8bee620", | ||||
|         "type": "github" | ||||
|       }, | ||||
|       "original": { | ||||
|   | ||||
							
								
								
									
										9
									
								
								metadata/en-US/changelogs/42.txt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								metadata/en-US/changelogs/42.txt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,9 @@ | ||||
| * Private messages interface overhaul in progress. | ||||
| * Added a loading indicator. | ||||
| * Documented the core JavaScript. | ||||
| * Fixed @-completion. | ||||
| * Covered up launch on Android with the splash screen. | ||||
| * Update: | ||||
|   * CodeMirror | ||||
|   * OpenSSL 3.5.2 | ||||
|   * speedscope 1.23.1 | ||||
							
								
								
									
										8
									
								
								metadata/en-US/changelogs/43.txt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										8
									
								
								metadata/en-US/changelogs/43.txt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,8 @@ | ||||
| * Fixed multiple issues with blob replication. | ||||
| * Fixed some link encoding issues. | ||||
| * Fixed some context menus being cut off. | ||||
| * Minor Android fixes. | ||||
| * Updates: | ||||
|   * CodeMirror | ||||
|   * OpenSSL 3.5.3 | ||||
|   * QuickJS 2025-09-13 | ||||
| @@ -1,8 +1,8 @@ | ||||
| <?xml version="1.0" encoding="utf-8"?> | ||||
| <manifest xmlns:android="http://schemas.android.com/apk/res/android" | ||||
| 	package="com.unprompted.tildefriends" | ||||
| 	android:versionCode="41" | ||||
| 	android:versionName="0.2025.8-wip"> | ||||
| 	android:versionCode="43" | ||||
| 	android:versionName="0.2025.9"> | ||||
| 	<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE"/> | ||||
| 	<uses-permission android:name="android.permission.INTERNET"/> | ||||
| 	<application | ||||
|   | ||||
| @@ -19,12 +19,12 @@ import android.os.RemoteException; | ||||
| import android.os.StrictMode; | ||||
| import android.text.InputType; | ||||
| import android.util.Base64; | ||||
| import android.util.Log; | ||||
| import android.view.KeyEvent; | ||||
| import android.view.MotionEvent; | ||||
| import android.view.View; | ||||
| import android.view.ViewGroup.LayoutParams; | ||||
| import android.view.Window; | ||||
| import android.view.ViewTreeObserver; | ||||
| import android.webkit.CookieManager; | ||||
| import android.webkit.DownloadListener; | ||||
| import android.webkit.JsPromptResult; | ||||
| @@ -43,6 +43,7 @@ import java.io.File; | ||||
| import java.io.FileOutputStream; | ||||
| import java.io.FileReader; | ||||
| import java.io.OutputStream; | ||||
| import java.util.concurrent.LinkedBlockingQueue; | ||||
| import java.util.concurrent.TimeUnit; | ||||
|  | ||||
| public class TildeFriendsActivity extends Activity { | ||||
| @@ -52,28 +53,42 @@ public class TildeFriendsActivity extends Activity { | ||||
| 	String port_file_path; | ||||
| 	Thread create_thread; | ||||
| 	Thread server_thread; | ||||
| 	Thread log_thread; | ||||
| 	ServiceConnection service_connection; | ||||
| 	FileObserver observer; | ||||
| 	LinkedBlockingQueue<String> log_queue = new LinkedBlockingQueue<String>(); | ||||
|  | ||||
| 	private ValueCallback<Uri[]> upload_message; | ||||
| 	private final static int FILECHOOSER_RESULT = 1; | ||||
| 	private float touch_down_y; | ||||
| 	private boolean ready = false; | ||||
| 	private boolean loaded = false; | ||||
| 	private boolean shutting_down = false; | ||||
|  | ||||
| 	static { | ||||
| 		Log.w("tildefriends", "Calling system.loadLibrary()."); | ||||
| 		log("Calling system.loadLibrary()."); | ||||
| 		System.loadLibrary("tildefriends"); | ||||
| 		Log.w("tildefriends", "system.loadLibrary() completed."); | ||||
| 		log("system.loadLibrary() completed."); | ||||
| 	} | ||||
|  | ||||
| 	public static native int tf_server_main(String files_dir, String apk_path, String out_port_file_path, ConnectivityManager connectivity_manager); | ||||
| 	public static native int tf_sandbox_main(int pipe_fd); | ||||
|  | ||||
| 	public static void log(String message) { | ||||
| 		if (s_activity != null && s_activity.log_queue != null && message != null) { | ||||
| 			try { | ||||
| 				s_activity.log_queue.put(message); | ||||
| 			} catch (InterruptedException e) { | ||||
| 				android.util.Log.w("tildefriends", message); | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	private void createThread() { | ||||
| 		web_view = (TildeFriendsWebView)findViewById(R.id.web); | ||||
| 		set_status("Extracting executable..."); | ||||
| 		Log.w("tildefriends", String.format("getFilesDir() is %s", getFilesDir().toString())); | ||||
| 		Log.w("tildefriends", String.format("getPackageResourcePath() is %s", getPackageResourcePath().toString())); | ||||
| 		Log.w("tildefriends", String.format("nativeLibraryDir is %s", getApplicationInfo().nativeLibraryDir)); | ||||
| 		log(String.format("getFilesDir() is %s", getFilesDir().toString())); | ||||
| 		log(String.format("getPackageResourcePath() is %s", getPackageResourcePath().toString())); | ||||
| 		log(String.format("nativeLibraryDir is %s", getApplicationInfo().nativeLibraryDir)); | ||||
|  | ||||
| 		port_file_path = getFilesDir().toString() + "/port.txt"; | ||||
| 		new File(port_file_path).delete(); | ||||
| @@ -81,21 +96,20 @@ public class TildeFriendsActivity extends Activity { | ||||
|  | ||||
| 		TildeFriendsActivity activity = this; | ||||
|  | ||||
| 		set_status("Starting server..."); | ||||
| 		server_thread = new Thread(new Runnable() { | ||||
| 			@Override | ||||
| 			public void run() { | ||||
| 				Log.w("tildefriends", "Watching for changes in: " + getFilesDir().toString()); | ||||
| 				log("Watching for changes in: " + getFilesDir().toString()); | ||||
| 				observer = make_file_observer(getFilesDir().toString(), port_file_path); | ||||
| 				observer.startWatching(); | ||||
|  | ||||
| 				Log.w("tildefriends", "Calling tf_server_main."); | ||||
| 				log("Calling tf_server_main."); | ||||
| 				int result = tf_server_main( | ||||
| 					getFilesDir().toString(), | ||||
| 					getPackageResourcePath().toString(), | ||||
| 					port_file_path, | ||||
| 					(ConnectivityManager)getApplicationContext().getSystemService(CONNECTIVITY_SERVICE)); | ||||
| 				Log.w("tildefriends", "tf_server_main returned " + result + "."); | ||||
| 				log("tf_server_main returned " + result + "."); | ||||
| 			} | ||||
| 		}); | ||||
| 		server_thread.start(); | ||||
| @@ -109,17 +123,17 @@ public class TildeFriendsActivity extends Activity { | ||||
|  | ||||
| 			web_view.setDownloadListener(new DownloadListener() { | ||||
| 				public void onDownloadStart(String url, String userAgent, String content_disposition, String mime_type, long content_length) { | ||||
| 					Log.w("tildefriends", "Let's download: " + url + " (" + content_disposition + ")"); | ||||
| 					log("Let's download: " + url + " (" + content_disposition + ")"); | ||||
| 					String file_name = URLUtil.guessFileName(url, content_disposition, mime_type); | ||||
| 					if (url.startsWith("data:") && url.indexOf(',') != -1) { | ||||
| 						String b64 = url.substring(url.indexOf(',') + 1); | ||||
| 						byte[] data = Base64.decode(b64, Base64.DEFAULT); | ||||
| 						Log.w("tildefriends", "Downloaded " + data.length + " bytes."); | ||||
| 						log("Downloaded " + data.length + " bytes."); | ||||
| 						File path = Environment.getExternalStoragePublicDirectory(Environment.DIRECTORY_DOWNLOADS); | ||||
| 						try (OutputStream stream = new FileOutputStream(new File(path, file_name))) { | ||||
| 							stream.write(data); | ||||
| 						} catch (java.io.IOException e) { | ||||
| 							Log.w("tildefriends", "IOException: " + e.toString()); | ||||
| 							log("IOException: " + e.toString()); | ||||
| 						} | ||||
| 						Toast.makeText(getApplicationContext(), "Downloaded File", Toast.LENGTH_LONG).show(); | ||||
| 					} else { | ||||
| @@ -227,12 +241,13 @@ public class TildeFriendsActivity extends Activity { | ||||
|  | ||||
| 				@Override | ||||
| 				public boolean onConsoleMessage(android.webkit.ConsoleMessage consoleMessage) { | ||||
| 					Log.d("tildefriends", consoleMessage.message() + " -- From line " + consoleMessage.lineNumber() + " of " + consoleMessage.sourceId()); | ||||
| 					log(consoleMessage.message() + " -- From line " + consoleMessage.lineNumber() + " of " + consoleMessage.sourceId()); | ||||
| 					return true; | ||||
| 				} | ||||
| 			}); | ||||
|  | ||||
| 			web_view.setWebViewClient(new WebViewClient() { | ||||
| 				@Override | ||||
| 				public boolean shouldOverrideUrlLoading(WebView view, WebResourceRequest request) | ||||
| 				{ | ||||
| 					if (request.getUrl() != null && request.getUrl().toString().startsWith(base_url)) { | ||||
| @@ -242,6 +257,11 @@ public class TildeFriendsActivity extends Activity { | ||||
| 						return true; | ||||
| 					} | ||||
| 				} | ||||
|  | ||||
| 				@Override | ||||
| 				public void onPageFinished(WebView view, String url) { | ||||
| 					s_activity.loaded = true; | ||||
| 				} | ||||
| 			}); | ||||
| 		}); | ||||
|  | ||||
| @@ -252,6 +272,23 @@ public class TildeFriendsActivity extends Activity { | ||||
| 	protected void onCreate(Bundle savedInstanceState) { | ||||
| 		s_activity = this; | ||||
| 		super.onCreate(savedInstanceState); | ||||
| 		log_thread = new Thread(new Runnable() { | ||||
| 			@Override | ||||
| 			public void run() { | ||||
| 				while (!s_activity.shutting_down) { | ||||
| 					try { | ||||
| 						String message = log_queue.take(); | ||||
| 						if (message != null) { | ||||
| 							android.util.Log.w("tildefriends", message); | ||||
| 						} else { | ||||
| 							break; | ||||
| 						} | ||||
| 					} catch (InterruptedException e) { | ||||
| 					} | ||||
| 				} | ||||
| 			} | ||||
| 		}); | ||||
| 		log_thread.start(); | ||||
| 		StrictMode.setThreadPolicy( | ||||
| 			new StrictMode.ThreadPolicy.Builder() | ||||
| 				.detectAll() | ||||
| @@ -271,6 +308,21 @@ public class TildeFriendsActivity extends Activity { | ||||
| 		refresh.setVisibility(View.GONE); | ||||
| 		refresh.setText("REFRESH"); | ||||
|  | ||||
| 		final View content = findViewById(android.R.id.content); | ||||
| 		content.getViewTreeObserver().addOnPreDrawListener( | ||||
| 			new ViewTreeObserver.OnPreDrawListener() { | ||||
| 				@Override | ||||
| 				public boolean onPreDraw() { | ||||
| 					if (s_activity.ready && s_activity.loaded) { | ||||
| 						content.getViewTreeObserver().removeOnPreDrawListener(this); | ||||
| 						return true; | ||||
| 					} else { | ||||
| 						return false; | ||||
| 					} | ||||
| 				} | ||||
| 			} | ||||
| 		); | ||||
|  | ||||
| 		create_thread = new Thread(new Runnable() { | ||||
| 			@Override | ||||
| 			public void run() { | ||||
| @@ -283,8 +335,17 @@ public class TildeFriendsActivity extends Activity { | ||||
| 	@Override | ||||
| 	protected void onDestroy() | ||||
| 	{ | ||||
| 		super.onDestroy(); | ||||
| 		try { | ||||
| 			shutting_down = true; | ||||
| 			if (log_queue != null) { | ||||
| 					log_queue.put("Goodbye."); | ||||
| 			} | ||||
| 			log_thread.join(); | ||||
| 		} catch (InterruptedException e) { | ||||
| 		} | ||||
| 		log_thread = null; | ||||
| 		s_activity = null; | ||||
| 		super.onDestroy(); | ||||
| 	} | ||||
|  | ||||
| 	@Override | ||||
| @@ -376,46 +437,33 @@ public class TildeFriendsActivity extends Activity { | ||||
| 		return -1; | ||||
| 	} | ||||
|  | ||||
| 	private void set_status(String text) { | ||||
| 		TextView text_view = (TextView)findViewById(R.id.text); | ||||
| 		web_view.setVisibility(View.GONE); | ||||
| 		text_view.setVisibility(View.VISIBLE); | ||||
| 		text_view.setText(text); | ||||
| 	} | ||||
|  | ||||
| 	private void hide_status() { | ||||
| 		TextView text_view = (TextView)findViewById(R.id.text); | ||||
| 		web_view.setVisibility(View.VISIBLE); | ||||
| 		text_view.setVisibility(View.GONE); | ||||
| 	} | ||||
|  | ||||
| 	public static void start_sandbox(int pipe_fd) { | ||||
| 		Log.w("tildefriends", "starting service with fd: " + pipe_fd); | ||||
| 		log("starting service with fd: " + pipe_fd); | ||||
| 		Intent intent = new Intent(s_activity, TildeFriendsSandboxService.class); | ||||
| 		s_activity.service_connection = new ServiceConnection() { | ||||
| 			@Override | ||||
| 			public void onBindingDied(ComponentName name) { | ||||
| 				Log.w("tildefriends", "onBindingDied"); | ||||
| 				log("onBindingDied"); | ||||
| 			} | ||||
|  | ||||
| 			@Override | ||||
| 			public void onNullBinding(ComponentName name) { | ||||
| 				Log.w("tildefriends", "onNullBinding"); | ||||
| 				log("onNullBinding"); | ||||
| 			} | ||||
|  | ||||
| 			@Override | ||||
| 			public void onServiceConnected(ComponentName name, IBinder binder) { | ||||
| 				Log.w("tildefriends", "onServiceConnected"); | ||||
| 				log("onServiceConnected"); | ||||
| 				Parcel data = Parcel.obtain(); | ||||
| 				try (ParcelFileDescriptor pfd = ParcelFileDescriptor.fromFd(pipe_fd)) { | ||||
| 					data.writeParcelable(pfd, 0); | ||||
| 				} catch (java.io.IOException e) { | ||||
| 					Log.w("tildefriends", "IOException: " + e); | ||||
| 					log("IOException: " + e); | ||||
| 				} | ||||
| 				try { | ||||
| 					binder.transact(TildeFriendsSandboxService.START_CALL,  data, null, IBinder.FLAG_ONEWAY); | ||||
| 				} catch (RemoteException e) { | ||||
| 					Log.w("tildefriends", "RemoteException"); | ||||
| 					log("RemoteException"); | ||||
| 				} finally { | ||||
| 					data.recycle(); | ||||
| 				} | ||||
| @@ -423,14 +471,14 @@ public class TildeFriendsActivity extends Activity { | ||||
|  | ||||
| 			@Override | ||||
| 			public void onServiceDisconnected(ComponentName name) { | ||||
| 				Log.w("tildefriends", "onServiceDisconnected"); | ||||
| 				log("onServiceDisconnected"); | ||||
| 			} | ||||
| 		}; | ||||
| 		s_activity.bindService(intent, s_activity.service_connection, BIND_AUTO_CREATE | BIND_NOT_FOREGROUND); | ||||
| 	} | ||||
|  | ||||
| 	public static void stop_sandbox() { | ||||
| 		Log.w("tildefriends", "stop_sandbox"); | ||||
| 		log("stop_sandbox"); | ||||
| 		if (s_activity.service_connection != null) { | ||||
| 			s_activity.unbindService(s_activity.service_connection); | ||||
| 			s_activity.service_connection = null; | ||||
| @@ -442,15 +490,11 @@ public class TildeFriendsActivity extends Activity { | ||||
| 		if (port >= 0) { | ||||
| 			base_url = "http://127.0.0.1:" + String.valueOf(port) + "/"; | ||||
| 			runOnUiThread(() -> { | ||||
| 				hide_status(); | ||||
| 				ready = true; | ||||
| 				web_view.loadUrl(base_url + "login/auto"); | ||||
| 			}); | ||||
| 			observer.stopWatching(); | ||||
| 			observer = null; | ||||
| 		} else { | ||||
| 			runOnUiThread(() -> { | ||||
| 				set_status("Waiting to connect..."); | ||||
| 			}); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
|   | ||||
| @@ -6,7 +6,6 @@ import android.os.Binder; | ||||
| import android.os.IBinder; | ||||
| import android.os.Parcel; | ||||
| import android.os.ParcelFileDescriptor; | ||||
| import android.util.Log; | ||||
|  | ||||
| public class TildeFriendsSandboxService extends Service { | ||||
| 	public static final int START_CALL = IBinder.FIRST_CALL_TRANSACTION; | ||||
| @@ -14,12 +13,12 @@ public class TildeFriendsSandboxService extends Service { | ||||
| 	Thread thread; | ||||
|  | ||||
| 	public int onStartCommand(Intent intent, int flags, int start_id) { | ||||
| 		Log.w("tildefriends", "TildeFriendsSandboxService: onStartCommand"); | ||||
| 		TildeFriendsActivity.log("TildeFriendsSandboxService: onStartCommand"); | ||||
| 		return super.onStartCommand(intent, flags, start_id); | ||||
| 	} | ||||
|  | ||||
| 	public void onDestroy() { | ||||
| 		Log.w("tildefriends", "TildeFriendsSandboxService: onDestroy"); | ||||
| 		TildeFriendsActivity.log("TildeFriendsSandboxService: onDestroy"); | ||||
| 		super.onDestroy(); | ||||
| 	} | ||||
|  | ||||
| @@ -27,9 +26,9 @@ public class TildeFriendsSandboxService extends Service { | ||||
| 		thread = new Thread(new Runnable() { | ||||
| 			@Override | ||||
| 			public void run() { | ||||
| 				Log.w("tildefriends", "Calling tf_sandbox_main."); | ||||
| 				TildeFriendsActivity.log("Calling tf_sandbox_main."); | ||||
| 				int result = TildeFriendsActivity.tf_sandbox_main(pipe_fd); | ||||
| 				Log.w("tildefriends", "tf_sandbox_main returned " + result + "."); | ||||
| 				TildeFriendsActivity.log("tf_sandbox_main returned " + result + "."); | ||||
| 			} | ||||
| 		}); | ||||
| 		thread.start(); | ||||
| @@ -43,7 +42,7 @@ public class TildeFriendsSandboxService extends Service { | ||||
| 				if (code == START_CALL) { | ||||
| 					ParcelFileDescriptor pfd = read_pfd(data); | ||||
| 					if (pfd != null) { | ||||
| 						Log.w("tildefriends", "fd is " + pfd.getFd()); | ||||
| 						TildeFriendsActivity.log("fd is " + pfd.getFd()); | ||||
| 						start_thread(pfd.detachFd()); | ||||
| 						try { | ||||
| 							pfd.close(); | ||||
|   | ||||
| @@ -10,11 +10,6 @@ | ||||
| 		android:id="@+id/web" | ||||
| 		android:layout_width="match_parent" | ||||
| 		android:layout_height="match_parent"/> | ||||
| 	<TextView | ||||
| 		android:id="@+id/text" | ||||
| 		android:layout_width="match_parent" | ||||
| 		android:layout_height="match_parent" | ||||
| 		android:gravity="center_horizontal|center_vertical"/> | ||||
| 	<TextView | ||||
| 		android:id="@+id/refresh" | ||||
| 		android:layout_width="match_parent" | ||||
|   | ||||
							
								
								
									
										65
									
								
								src/api.js.c
									
									
									
									
									
								
							
							
						
						
									
										65
									
								
								src/api.js.c
									
									
									
									
									
								
							| @@ -147,12 +147,77 @@ static JSValue _tf_api_core_apps(JSContext* context, JSValueConst this_val, int | ||||
| 	return result; | ||||
| } | ||||
|  | ||||
| static JSValue _tf_api_core_register(JSContext* context, JSValueConst this_val, int argc, JSValueConst* argv, int magic, JSValue* data) | ||||
| { | ||||
| 	JSValue event_name = argv[0]; | ||||
| 	JSValue handler = argv[1]; | ||||
| 	JSValue process = data[0]; | ||||
| 	JSValue event_handlers = JS_GetPropertyStr(context, process, "eventHandlers"); | ||||
| 	JSAtom atom = JS_ValueToAtom(context, event_name); | ||||
| 	JSValue array = JS_GetProperty(context, event_handlers, atom); | ||||
| 	if (!JS_IsArray(context, array)) | ||||
| 	{ | ||||
| 		JS_FreeValue(context, array); | ||||
| 		array = JS_NewArray(context); | ||||
| 		JS_SetProperty(context, event_handlers, atom, JS_DupValue(context, array)); | ||||
| 	} | ||||
| 	JS_SetPropertyUint32(context, array, tf_util_get_length(context, array), JS_DupValue(context, handler)); | ||||
| 	JS_FreeValue(context, array); | ||||
| 	JS_FreeAtom(context, atom); | ||||
| 	JS_FreeValue(context, event_handlers); | ||||
| 	return JS_UNDEFINED; | ||||
| } | ||||
|  | ||||
| static JSValue _tf_api_core_unregister(JSContext* context, JSValueConst this_val, int argc, JSValueConst* argv, int magic, JSValue* data) | ||||
| { | ||||
| 	JSValue event_name = argv[0]; | ||||
| 	JSValue handler = argv[1]; | ||||
| 	JSValue process = data[0]; | ||||
| 	JSValue event_handlers = JS_GetPropertyStr(context, process, "eventHandlers"); | ||||
| 	JSAtom atom = JS_ValueToAtom(context, event_name); | ||||
| 	JSValue array = JS_GetProperty(context, event_handlers, atom); | ||||
|  | ||||
| 	if (JS_IsArray(context, array)) | ||||
| 	{ | ||||
| 		JSValue index_of = JS_GetPropertyStr(context, array, "indexOf"); | ||||
|  | ||||
| 		JSValue index = JS_Call(context, index_of, array, 1, &handler); | ||||
| 		int int_index = -1; | ||||
| 		JS_ToInt32(context, &int_index, index); | ||||
| 		if (int_index != -1) | ||||
| 		{ | ||||
| 			JSValue splice = JS_GetPropertyStr(context, array, "splice"); | ||||
| 			JSValue splice_args[] = { | ||||
| 				index, | ||||
| 				JS_NewInt32(context, 1), | ||||
| 			}; | ||||
| 			JSValue result = JS_Call(context, splice, array, 2, splice_args); | ||||
| 			JS_FreeValue(context, result); | ||||
| 			JS_FreeValue(context, splice); | ||||
| 		} | ||||
| 		JS_FreeValue(context, index); | ||||
| 		JS_FreeValue(context, index_of); | ||||
|  | ||||
| 		if (tf_util_get_length(context, array) == 0) | ||||
| 		{ | ||||
| 			JS_DeleteProperty(context, event_handlers, atom, 0); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	JS_FreeValue(context, array); | ||||
| 	JS_FreeAtom(context, atom); | ||||
| 	JS_FreeValue(context, event_handlers); | ||||
| 	return JS_UNDEFINED; | ||||
| } | ||||
|  | ||||
| static JSValue _tf_api_register_imports(JSContext* context, JSValueConst this_val, int argc, JSValueConst* argv) | ||||
| { | ||||
| 	JSValue imports = argv[0]; | ||||
| 	JSValue process = argv[1]; | ||||
| 	JSValue core = JS_GetPropertyStr(context, imports, "core"); | ||||
| 	JS_SetPropertyStr(context, core, "apps", JS_NewCFunctionData(context, _tf_api_core_apps, 1, 0, 1, &process)); | ||||
| 	JS_SetPropertyStr(context, core, "register", JS_NewCFunctionData(context, _tf_api_core_register, 2, 0, 1, &process)); | ||||
| 	JS_SetPropertyStr(context, core, "unregister", JS_NewCFunctionData(context, _tf_api_core_unregister, 2, 0, 1, &process)); | ||||
| 	JS_FreeValue(context, core); | ||||
| 	return JS_UNDEFINED; | ||||
| } | ||||
|   | ||||
| @@ -13,13 +13,13 @@ | ||||
| 	<key>CFBundlePackageType</key> | ||||
| 	<string>APPL</string> | ||||
| 	<key>CFBundleShortVersionString</key> | ||||
| 	<string>0.2025.8</string> | ||||
| 	<string>0.2025.9</string> | ||||
| 	<key>CFBundleSupportedPlatforms</key> | ||||
| 	<array> | ||||
| 		<string>iPhoneOS</string> | ||||
| 	</array> | ||||
| 	<key>CFBundleVersion</key> | ||||
| 	<string>16</string> | ||||
| 	<string>17</string> | ||||
| 	<key>DTPlatformName</key> | ||||
| 	<string>iphoneos</string> | ||||
| 	<key>LSRequiresIPhoneOS</key> | ||||
|   | ||||
							
								
								
									
										18
									
								
								src/main.c
									
									
									
									
									
								
							
							
						
						
									
										18
									
								
								src/main.c
									
									
									
									
									
								
							| @@ -1578,28 +1578,28 @@ static void _shed_privileges() | ||||
| 	if (setrlimit(RLIMIT_FSIZE, &zeroLimit) != 0) | ||||
| 	{ | ||||
| 		perror("setrlimit(RLIMIT_FSIZE, {0, 0})"); | ||||
| 		exit(-1); | ||||
| 		exit(EXIT_FAILURE); | ||||
| 	} | ||||
| 	if (setrlimit(RLIMIT_NOFILE, &zeroLimit) != 0) | ||||
| 	{ | ||||
| 		perror("setrlimit(RLIMIT_NOFILE, {0, 0})"); | ||||
| 		exit(-1); | ||||
| 		exit(EXIT_FAILURE); | ||||
| 	} | ||||
| 	if (setrlimit(RLIMIT_NPROC, &zeroLimit) != 0) | ||||
| 	{ | ||||
| 		perror("setrlimit(RLIMIT_NPROC, {0, 0})"); | ||||
| 		exit(-1); | ||||
| 		exit(EXIT_FAILURE); | ||||
| 	} | ||||
| #if !defined(__MACH__) && !defined(__OpenBSD__) | ||||
| 	if (setrlimit(RLIMIT_LOCKS, &zeroLimit) != 0) | ||||
| 	{ | ||||
| 		perror("setrlimit(RLIMIT_LOCKS, {0, 0})"); | ||||
| 		exit(-1); | ||||
| 		exit(EXIT_FAILURE); | ||||
| 	} | ||||
| 	if (setrlimit(RLIMIT_MSGQUEUE, &zeroLimit) != 0) | ||||
| 	{ | ||||
| 		perror("setrlimit(RLIMIT_MSGQUEUE, {0, 0})"); | ||||
| 		exit(-1); | ||||
| 		exit(EXIT_FAILURE); | ||||
| 	} | ||||
| #endif | ||||
| #endif | ||||
| @@ -1609,12 +1609,12 @@ static void _shed_privileges() | ||||
| 	if (unveil("/dev/null", "r") || unveil(NULL, NULL)) | ||||
| 	{ | ||||
| 		perror("unveil"); | ||||
| 		exit(-1); | ||||
| 		exit(EXIT_FAILURE); | ||||
| 	} | ||||
| 	if (pledge("stdio unveil", NULL)) | ||||
| 	{ | ||||
| 		perror("pledge"); | ||||
| 		exit(-1); | ||||
| 		exit(EXIT_FAILURE); | ||||
| 	} | ||||
| #endif | ||||
| } | ||||
| @@ -1831,7 +1831,7 @@ static void _error_handler(int sig) | ||||
| 	const char* stack = tf_util_backtrace_string(); | ||||
| 	tf_printf("ERROR:\n%s\n", stack); | ||||
| 	tf_free((void*)stack); | ||||
| 	_exit(1); | ||||
| 	_exit(EXIT_FAILURE); | ||||
| } | ||||
|  | ||||
| #if defined(_WIN32) | ||||
| @@ -1843,7 +1843,7 @@ static LONG WINAPI _win32_exception_handler(EXCEPTION_POINTERS* info) | ||||
| 		const char* stack = tf_util_backtrace_string(); | ||||
| 		tf_printf("ERROR:\n%s\n", stack); | ||||
| 		tf_free((void*)stack); | ||||
| 		_exit(1); | ||||
| 		_exit(EXIT_FAILURE); | ||||
| 	} | ||||
| 	return EXCEPTION_CONTINUE_SEARCH; | ||||
| } | ||||
|   | ||||
							
								
								
									
										87
									
								
								src/ssb.c
									
									
									
									
									
								
							
							
						
						
									
										87
									
								
								src/ssb.c
									
									
									
									
									
								
							| @@ -133,6 +133,15 @@ typedef struct _tf_ssb_message_added_callback_node_t | ||||
| 	tf_ssb_message_added_callback_node_t* next; | ||||
| } tf_ssb_message_added_callback_node_t; | ||||
|  | ||||
| typedef struct _tf_ssb_blob_stored_callback_node_t tf_ssb_blob_stored_callback_node_t; | ||||
| typedef struct _tf_ssb_blob_stored_callback_node_t | ||||
| { | ||||
| 	tf_ssb_blob_stored_callback_t* callback; | ||||
| 	tf_ssb_callback_cleanup_t* cleanup; | ||||
| 	void* user_data; | ||||
| 	tf_ssb_blob_stored_callback_node_t* next; | ||||
| } tf_ssb_blob_stored_callback_node_t; | ||||
|  | ||||
| typedef struct _tf_ssb_blob_want_added_callback_node_t tf_ssb_blob_want_added_callback_node_t; | ||||
| typedef struct _tf_ssb_blob_want_added_callback_node_t | ||||
| { | ||||
| @@ -235,6 +244,9 @@ typedef struct _tf_ssb_t | ||||
| 	tf_ssb_message_added_callback_node_t* message_added; | ||||
| 	int message_added_count; | ||||
|  | ||||
| 	tf_ssb_blob_stored_callback_node_t* blob_stored; | ||||
| 	int blob_stored_count; | ||||
|  | ||||
| 	tf_ssb_blob_want_added_callback_node_t* blob_want_added; | ||||
| 	int blob_want_added_count; | ||||
|  | ||||
| @@ -2741,6 +2753,17 @@ void tf_ssb_destroy(tf_ssb_t* ssb) | ||||
| 		} | ||||
| 		tf_free(node); | ||||
| 	} | ||||
| 	while (ssb->blob_stored) | ||||
| 	{ | ||||
| 		tf_ssb_blob_stored_callback_node_t* node = ssb->blob_stored; | ||||
| 		ssb->blob_stored = node->next; | ||||
| 		ssb->blob_stored_count--; | ||||
| 		if (node->cleanup) | ||||
| 		{ | ||||
| 			node->cleanup(ssb, node->user_data); | ||||
| 		} | ||||
| 		tf_free(node); | ||||
| 	} | ||||
| 	while (ssb->blob_want_added) | ||||
| 	{ | ||||
| 		tf_ssb_blob_want_added_callback_node_t* node = ssb->blob_want_added; | ||||
| @@ -2783,6 +2806,16 @@ void tf_ssb_destroy(tf_ssb_t* ssb) | ||||
|  | ||||
| 	uv_run(ssb->loop, UV_RUN_NOWAIT); | ||||
|  | ||||
| 	if (ssb->own_context) | ||||
| 	{ | ||||
| 		if (!ssb->quiet) | ||||
| 		{ | ||||
| 			tf_printf("closing ssb context\n"); | ||||
| 		} | ||||
| 		JS_FreeContext(ssb->context); | ||||
| 		JS_FreeRuntime(ssb->runtime); | ||||
| 		ssb->own_context = false; | ||||
| 	} | ||||
| 	if (ssb->loop == &ssb->own_loop) | ||||
| 	{ | ||||
| 		if (!ssb->quiet) | ||||
| @@ -2799,16 +2832,6 @@ void tf_ssb_destroy(tf_ssb_t* ssb) | ||||
| 	{ | ||||
| 		tf_printf("uv loop closed.\n"); | ||||
| 	} | ||||
| 	if (ssb->own_context) | ||||
| 	{ | ||||
| 		if (!ssb->quiet) | ||||
| 		{ | ||||
| 			tf_printf("closing ssb context\n"); | ||||
| 		} | ||||
| 		JS_FreeContext(ssb->context); | ||||
| 		JS_FreeRuntime(ssb->runtime); | ||||
| 		ssb->own_context = false; | ||||
| 	} | ||||
| 	while (ssb->broadcasts) | ||||
| 	{ | ||||
| 		tf_ssb_broadcast_t* broadcast = ssb->broadcasts; | ||||
| @@ -3960,9 +3983,53 @@ void tf_ssb_remove_message_added_callback(tf_ssb_t* ssb, tf_ssb_message_added_ca | ||||
| 	} | ||||
| } | ||||
|  | ||||
| void tf_ssb_add_blob_stored_callback(tf_ssb_t* ssb, tf_ssb_blob_stored_callback_t* callback, void (*cleanup)(tf_ssb_t* ssb, void* user_data), void* user_data) | ||||
| { | ||||
| 	tf_ssb_blob_stored_callback_node_t* node = tf_malloc(sizeof(tf_ssb_blob_stored_callback_node_t)); | ||||
| 	*node = (tf_ssb_blob_stored_callback_node_t) { | ||||
| 		.callback = callback, | ||||
| 		.cleanup = cleanup, | ||||
| 		.user_data = user_data, | ||||
| 		.next = ssb->blob_stored, | ||||
| 	}; | ||||
| 	ssb->blob_stored = node; | ||||
| 	ssb->blob_stored_count++; | ||||
| } | ||||
|  | ||||
| void tf_ssb_remove_blob_stored_callback(tf_ssb_t* ssb, tf_ssb_blob_stored_callback_t* callback, void* user_data) | ||||
| { | ||||
| 	tf_ssb_blob_stored_callback_node_t** it = &ssb->blob_stored; | ||||
| 	while (*it) | ||||
| 	{ | ||||
| 		if ((*it)->callback == callback && (*it)->user_data == user_data) | ||||
| 		{ | ||||
| 			tf_ssb_blob_stored_callback_node_t* node = *it; | ||||
| 			*it = node->next; | ||||
| 			ssb->blob_stored_count--; | ||||
| 			if (node->cleanup) | ||||
| 			{ | ||||
| 				node->cleanup(ssb, node->user_data); | ||||
| 			} | ||||
| 			tf_free(node); | ||||
| 		} | ||||
| 		else | ||||
| 		{ | ||||
| 			it = &(*it)->next; | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| void tf_ssb_notify_blob_stored(tf_ssb_t* ssb, const char* id) | ||||
| { | ||||
| 	tf_ssb_blob_stored_callback_node_t* next = NULL; | ||||
| 	ssb->blobs_stored++; | ||||
| 	for (tf_ssb_blob_stored_callback_node_t* node = ssb->blob_stored; node; node = next) | ||||
| 	{ | ||||
| 		next = node->next; | ||||
| 		tf_trace_begin(ssb->trace, "blob stored callback"); | ||||
| 		node->callback(ssb, id, node->user_data); | ||||
| 		tf_trace_end(ssb->trace); | ||||
| 	} | ||||
| } | ||||
|  | ||||
| void tf_ssb_notify_message_added(tf_ssb_t* ssb, const char* author, int32_t sequence, const char* id, JSValue message_keys) | ||||
|   | ||||
							
								
								
									
										24
									
								
								src/ssb.db.c
									
									
									
									
									
								
							
							
						
						
									
										24
									
								
								src/ssb.db.c
									
									
									
									
									
								
							| @@ -421,6 +421,7 @@ void tf_ssb_db_init(tf_ssb_t* ssb) | ||||
| 		_tf_ssb_db_exec(db, "COMMIT TRANSACTION"); | ||||
| 		tf_printf("Done.\n"); | ||||
| 	} | ||||
| 	_tf_ssb_db_exec(db, "DELETE FROM blob_wants_cache WHERE blob_wants_cache.id IN (SELECT blobs.id FROM blobs)"); | ||||
| 	if (!_tf_ssb_db_has_rows(db, "SELECT * FROM sqlite_schema WHERE type = 'index' AND name = 'blob_wants_cache_source_id_unique_index'")) | ||||
| 	{ | ||||
| 		tf_printf("Creating blob_wants_cache UNIQUE constraint.\n"); | ||||
| @@ -436,8 +437,11 @@ void tf_ssb_db_init(tf_ssb_t* ssb) | ||||
| 	_tf_ssb_db_exec(db, | ||||
| 		"CREATE TRIGGER IF NOT EXISTS messages_ai_blob_wants_cache AFTER INSERT ON messages_refs BEGIN " | ||||
| 		"INSERT INTO blob_wants_cache (source, id, timestamp) " | ||||
| 		"SELECT messages.id, new.ref, messages.timestamp FROM messages WHERE messages.id = new.message AND " | ||||
| 		"LENGTH(new.ref) = 52 AND new.ref LIKE '&%.sha256' " | ||||
| 		"SELECT messages.id, new.ref, messages.timestamp FROM messages " | ||||
| 		"JOIN blobs ON new.ref = blobs.id " | ||||
| 		"WHERE messages.id = new.message AND " | ||||
| 		"LENGTH(new.ref) = 52 AND new.ref LIKE '&%.sha256' AND " | ||||
| 		"blobs.content IS NULL " | ||||
| 		"ON CONFLICT (source, id) DO NOTHING; END"); | ||||
| 	_tf_ssb_db_exec(db, "DROP TRIGGER IF EXISTS blobs_refs_ai_blob_wants_cache"); | ||||
| 	_tf_ssb_db_exec(db, | ||||
| @@ -445,6 +449,8 @@ void tf_ssb_db_init(tf_ssb_t* ssb) | ||||
| 		"INSERT INTO blob_wants_cache (source, id, timestamp) " | ||||
| 		"SELECT messages.id, new.ref, messages.timestamp FROM messages " | ||||
| 		"JOIN blob_wants_cache bwc ON bwc.source = messages.id AND bwc.id = new.blob " | ||||
| 		"JOIN blobs ON bwc.id = blobs.id " | ||||
| 		"WHERE blobs.content IS NULL " | ||||
| 		"ON CONFLICT (source, id) DO NOTHING; END"); | ||||
| 	_tf_ssb_db_exec(db, | ||||
| 		"CREATE TRIGGER IF NOT EXISTS messages_ad_blob_wants_cache AFTER DELETE ON messages BEGIN " | ||||
| @@ -574,16 +580,15 @@ static int64_t _tf_ssb_db_store_message_raw(sqlite3* db, const char* id, const c | ||||
| 	return last_row_id; | ||||
| } | ||||
|  | ||||
| static char* _tf_ssb_db_get_message_blob_wants(tf_ssb_t* ssb, int64_t rowid) | ||||
| static char* _tf_ssb_db_get_message_blob_wants(sqlite3* db, int64_t rowid) | ||||
| { | ||||
| 	sqlite3* db = tf_ssb_acquire_db_reader(ssb); | ||||
| 	sqlite3_stmt* statement; | ||||
| 	char* result = NULL; | ||||
| 	size_t size = 0; | ||||
|  | ||||
| 	if (sqlite3_prepare_v2(db, | ||||
| 			"SELECT DISTINCT json.value FROM messages, json_tree(messages.content) AS json LEFT OUTER JOIN blobs ON json.value = blobs.id WHERE messages.rowid = ?1 AND " | ||||
| 			"json.value LIKE '&%.sha256' AND length(json.value) = ?2 AND blobs.content IS NULL", | ||||
| 			"length(json.value) = ?2 AND json.value LIKE '&%.sha256' AND blobs.content IS NULL", | ||||
| 			-1, &statement, NULL) == SQLITE_OK) | ||||
| 	{ | ||||
| 		if (sqlite3_bind_int64(statement, 1, rowid) == SQLITE_OK && sqlite3_bind_int(statement, 2, k_blob_id_len - 1) == SQLITE_OK) | ||||
| @@ -615,7 +620,6 @@ static char* _tf_ssb_db_get_message_blob_wants(tf_ssb_t* ssb, int64_t rowid) | ||||
| 	result = tf_realloc(result, size + 1); | ||||
| 	result[size] = '\0'; | ||||
|  | ||||
| 	tf_ssb_release_db_reader(ssb, db); | ||||
| 	return result; | ||||
| } | ||||
|  | ||||
| @@ -653,7 +657,7 @@ static void _tf_ssb_db_store_message_work(tf_ssb_t* ssb, void* user_data) | ||||
| 		if (last_row_id != -1) | ||||
| 		{ | ||||
| 			store->out_stored = true; | ||||
| 			store->out_blob_wants = _tf_ssb_db_get_message_blob_wants(ssb, last_row_id); | ||||
| 			store->out_blob_wants = _tf_ssb_db_get_message_blob_wants(db, last_row_id); | ||||
| 		} | ||||
| 		store = store->next; | ||||
| 	} | ||||
| @@ -903,10 +907,6 @@ void tf_ssb_db_add_blob_wants(sqlite3* db, const char* id) | ||||
| 			{ | ||||
| 				tf_printf("blob wants cache update failed: %s.\n", sqlite3_errmsg(db)); | ||||
| 			} | ||||
| 			else | ||||
| 			{ | ||||
| 				tf_printf("want: %s\n", id); | ||||
| 			} | ||||
| 		} | ||||
| 		sqlite3_finalize(statement); | ||||
| 	} | ||||
| @@ -976,7 +976,7 @@ static void _tf_ssb_db_blob_store_work(tf_ssb_t* ssb, void* user_data) | ||||
| static void _tf_ssb_db_blob_store_after_work(tf_ssb_t* ssb, int status, void* user_data) | ||||
| { | ||||
| 	blob_store_work_t* blob_work = user_data; | ||||
| 	if (status == 0 && *blob_work->id) | ||||
| 	if (status == 0 && *blob_work->id && blob_work->is_new) | ||||
| 	{ | ||||
| 		tf_ssb_notify_blob_stored(ssb, blob_work->id); | ||||
| 	} | ||||
|   | ||||
							
								
								
									
										25
									
								
								src/ssb.h
									
									
									
									
									
								
							
							
						
						
									
										25
									
								
								src/ssb.h
									
									
									
									
									
								
							| @@ -680,6 +680,31 @@ void tf_ssb_remove_message_added_callback(tf_ssb_t* ssb, tf_ssb_message_added_ca | ||||
| */ | ||||
| void tf_ssb_notify_message_added(tf_ssb_t* ssb, const char* author, int32_t sequence, const char* id, JSValue message_with_keys); | ||||
|  | ||||
| /** | ||||
| ** A callback called when a blob is added to the database. | ||||
| ** @param ssb The SSB instance. | ||||
| ** @param id The blob identifier. | ||||
| ** @param user_data The user data. | ||||
| */ | ||||
| typedef void(tf_ssb_blob_stored_callback_t)(tf_ssb_t* ssb, const char* id, void* user_data); | ||||
|  | ||||
| /** | ||||
| ** Register a callback called when a blob is added to the database. | ||||
| ** @param ssb The SSB instance. | ||||
| ** @param callback The callback function. | ||||
| ** @param cleanup A function to call when the callback is removed. | ||||
| ** @param user_data User data to pass to the callback. | ||||
| */ | ||||
| void tf_ssb_add_blob_stored_callback(tf_ssb_t* ssb, tf_ssb_blob_stored_callback_t* callback, tf_ssb_callback_cleanup_t* cleanup, void* user_data); | ||||
|  | ||||
| /** | ||||
| ** Remove a callback registered for when a blob is added to the database. | ||||
| ** @param ssb The SSB instance. | ||||
| ** @param callback The callback function. | ||||
| ** @param user_data User data registered with the callback. | ||||
| */ | ||||
| void tf_ssb_remove_blob_stored_callback(tf_ssb_t* ssb, tf_ssb_blob_stored_callback_t* callback, void* user_data); | ||||
|  | ||||
| /** | ||||
| ** Record that a new blob was stored. | ||||
| ** @param ssb The SSB instance. | ||||
|   | ||||
							
								
								
									
										24
									
								
								src/ssb.js.c
									
									
									
									
									
								
							
							
						
						
									
										24
									
								
								src/ssb.js.c
									
									
									
									
									
								
							| @@ -1627,6 +1627,20 @@ static void _tf_ssb_on_message_added_callback(tf_ssb_t* ssb, const char* author, | ||||
| 	JS_FreeValue(context, string); | ||||
| } | ||||
|  | ||||
| static void _tf_ssb_on_blob_stored_callback(tf_ssb_t* ssb, const char* id, void* user_data) | ||||
| { | ||||
| 	JSContext* context = tf_ssb_get_context(ssb); | ||||
| 	JSValue callback = JS_MKPTR(JS_TAG_OBJECT, user_data); | ||||
| 	JSValue string = JS_NewString(context, id); | ||||
| 	JSValue response = JS_Call(context, callback, JS_UNDEFINED, 1, &string); | ||||
| 	if (tf_util_report_error(context, response)) | ||||
| 	{ | ||||
| 		tf_ssb_remove_blob_stored_callback(ssb, _tf_ssb_on_blob_stored_callback, user_data); | ||||
| 	} | ||||
| 	JS_FreeValue(context, response); | ||||
| 	JS_FreeValue(context, string); | ||||
| } | ||||
|  | ||||
| static void _tf_ssb_on_blob_want_added_callback(tf_ssb_t* ssb, const char* id, void* user_data) | ||||
| { | ||||
| 	JSContext* context = tf_ssb_get_context(ssb); | ||||
| @@ -1747,6 +1761,11 @@ static JSValue _tf_ssb_add_event_listener(JSContext* context, JSValueConst this_ | ||||
| 			void* ptr = JS_VALUE_GET_PTR(JS_DupValue(context, callback)); | ||||
| 			tf_ssb_add_message_added_callback(ssb, _tf_ssb_on_message_added_callback, _tf_ssb_cleanup_value, ptr); | ||||
| 		} | ||||
| 		else if (strcmp(event_name, "blob") == 0) | ||||
| 		{ | ||||
| 			void* ptr = JS_VALUE_GET_PTR(JS_DupValue(context, callback)); | ||||
| 			tf_ssb_add_blob_stored_callback(ssb, _tf_ssb_on_blob_stored_callback, _tf_ssb_cleanup_value, ptr); | ||||
| 		} | ||||
| 		else if (strcmp(event_name, "blob_want_added") == 0) | ||||
| 		{ | ||||
| 			void* ptr = JS_VALUE_GET_PTR(JS_DupValue(context, callback)); | ||||
| @@ -1790,6 +1809,11 @@ static JSValue _tf_ssb_remove_event_listener(JSContext* context, JSValueConst th | ||||
| 			void* ptr = JS_VALUE_GET_PTR(JS_DupValue(context, callback)); | ||||
| 			tf_ssb_remove_message_added_callback(ssb, _tf_ssb_on_message_added_callback, ptr); | ||||
| 		} | ||||
| 		else if (strcmp(event_name, "blob") == 0) | ||||
| 		{ | ||||
| 			void* ptr = JS_VALUE_GET_PTR(JS_DupValue(context, callback)); | ||||
| 			tf_ssb_remove_blob_stored_callback(ssb, _tf_ssb_on_blob_stored_callback, ptr); | ||||
| 		} | ||||
| 		else if (strcmp(event_name, "blob_want_added") == 0) | ||||
| 		{ | ||||
| 			void* ptr = JS_VALUE_GET_PTR(JS_DupValue(context, callback)); | ||||
|   | ||||
| @@ -152,6 +152,12 @@ static void _wait_stored(tf_ssb_t* ssb, bool* stored) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| static void _blob_stored(tf_ssb_t* ssb, const char* id, void* user_data) | ||||
| { | ||||
| 	tf_printf("blob stored %s\n", id); | ||||
| 	*(bool*)user_data = true; | ||||
| } | ||||
|  | ||||
| void tf_ssb_test_ssb(const tf_test_options_t* options) | ||||
| { | ||||
| 	tf_printf("Testing SSB.\n"); | ||||
| @@ -224,8 +230,13 @@ void tf_ssb_test_ssb(const tf_test_options_t* options) | ||||
|  | ||||
| 	char blob_id[k_id_base64_len] = { 0 }; | ||||
| 	const char* k_blob = "Hello, blob!"; | ||||
| 	bool blob_stored = false; | ||||
| 	tf_ssb_add_blob_stored_callback(ssb0, _blob_stored, NULL, &blob_stored); | ||||
| 	b = tf_ssb_db_blob_store(ssb0, (const uint8_t*)k_blob, strlen(k_blob), blob_id, sizeof(blob_id), NULL); | ||||
| 	tf_ssb_notify_blob_stored(ssb0, blob_id); | ||||
| 	tf_ssb_remove_blob_stored_callback(ssb0, _blob_stored, &blob_stored); | ||||
| 	assert(b); | ||||
| 	assert(blob_stored); | ||||
|  | ||||
| 	JSContext* context0 = tf_ssb_get_context(ssb0); | ||||
| 	JSValue obj = JS_NewObject(context0); | ||||
| @@ -1117,6 +1128,11 @@ void tf_ssb_test_replicate(const tf_test_options_t* options) | ||||
| 		tf_printf("%s user %d = %s private=%s\n", added ? "added" : "failed", i, public[i], private[i]); | ||||
| 	} | ||||
|  | ||||
| 	char blob_id[k_id_base64_len] = { 0 }; | ||||
| 	const char* k_blob = "Hello, new blob!"; | ||||
| 	b = tf_ssb_db_blob_store(ssb0, (const uint8_t*)k_blob, strlen(k_blob), blob_id, sizeof(blob_id), NULL); | ||||
| 	assert(b); | ||||
|  | ||||
| 	JSContext* context0 = tf_ssb_get_context(ssb0); | ||||
| 	for (int i = 0; i < k_key_count - 1; i++) | ||||
| 	{ | ||||
| @@ -1141,6 +1157,7 @@ void tf_ssb_test_replicate(const tf_test_options_t* options) | ||||
| 		obj = JS_NewObject(context0); | ||||
| 		JS_SetPropertyStr(context0, obj, "type", JS_NewString(context0, "post")); | ||||
| 		JS_SetPropertyStr(context0, obj, "text", JS_NewString(context0, "Hello, world!")); | ||||
| 		JS_SetPropertyStr(context0, obj, "arbitrary_reference", JS_NewString(context0, blob_id)); | ||||
| 		stored = false; | ||||
| 		signed_message = tf_ssb_sign_message(ssb0, self, private_bin, obj, NULL, 0); | ||||
| 		tf_ssb_verify_strip_and_store_message(ssb0, signed_message, _message_stored, &stored); | ||||
| @@ -1202,6 +1219,15 @@ void tf_ssb_test_replicate(const tf_test_options_t* options) | ||||
| 	tf_ssb_remove_message_added_callback(ssb1, _message_added, &count1); | ||||
| 	tf_printf("done\n"); | ||||
|  | ||||
| 	tf_printf("Waiting for blob.\n"); | ||||
| 	while (!tf_ssb_db_blob_get(ssb0, blob_id, NULL, NULL)) | ||||
| 	{ | ||||
| 		tf_ssb_set_main_thread(ssb1, true); | ||||
| 		uv_run(&loop, UV_RUN_ONCE); | ||||
| 		tf_ssb_set_main_thread(ssb1, false); | ||||
| 	} | ||||
| 	tf_printf("done\n"); | ||||
|  | ||||
| 	tf_ssb_send_close(ssb1); | ||||
|  | ||||
| 	uv_close((uv_handle_t*)&idle0, NULL); | ||||
|   | ||||
| @@ -1101,7 +1101,7 @@ void tf_task_on_receive_packet(int packetType, const char* begin, size_t length, | ||||
| 		} | ||||
| 		else | ||||
| 		{ | ||||
| 			exit(1); | ||||
| 			exit(EXIT_FAILURE); | ||||
| 		} | ||||
| 		break; | ||||
| 	case kSetImports: | ||||
| @@ -1699,7 +1699,7 @@ void tf_task_activate(tf_task_t* task) | ||||
| 				else | ||||
| 				{ | ||||
| 					tf_printf("Assignment missing '=': %s.\n", assignment); | ||||
| 					exit(1); | ||||
| 					exit(EXIT_FAILURE); | ||||
| 				} | ||||
| 			} | ||||
| 			tf_free(copy); | ||||
|   | ||||
| @@ -1,2 +1,2 @@ | ||||
| #define VERSION_NUMBER "0.2025.8-wip" | ||||
| #define VERSION_NUMBER "0.2025.9" | ||||
| #define VERSION_NAME "This program kills fascists." | ||||
|   | ||||
| @@ -93,7 +93,7 @@ try: | ||||
| 	select(driver, ['#document', 'frame', '#gs_room_name'], ('send_keys', 'test room')) | ||||
| 	select(driver, ['#document', 'frame', '//*[@id="gs_room_name"]/following-sibling::button'], ('click',)) | ||||
| 	select(driver, ['//button[text()="✅ Allow"]'], ('click',)) | ||||
| 	driver.switch_to.alert.accept() | ||||
| 	wait.until(expected_conditions.alert_is_present()).accept() | ||||
|  | ||||
| 	select(driver, ['tf-navigation', 'shadow_root', '#identity'], ('click',)) | ||||
| 	select(driver, ['tf-navigation', 'shadow_root', '#logout'], ('click',)) | ||||
|   | ||||
		Reference in New Issue
	
	Block a user