mirror of
https://github.com/go-gitea/gitea.git
synced 2025-04-12 10:48:46 +03:00
Fix various Fomantic UI and htmx problems (#33851)
Also fix #31328, fix #33854
This commit is contained in:
@ -79,7 +79,6 @@ cpu.out
|
|||||||
/public/assets/fonts
|
/public/assets/fonts
|
||||||
/public/assets/img/avatar
|
/public/assets/img/avatar
|
||||||
/vendor
|
/vendor
|
||||||
/web_src/fomantic
|
|
||||||
/VERSION
|
/VERSION
|
||||||
/.air
|
/.air
|
||||||
/.go-licenses
|
/.go-licenses
|
||||||
|
@ -754,17 +754,9 @@ $.api = $.fn.api = function(parameters) {
|
|||||||
;
|
;
|
||||||
},
|
},
|
||||||
urlEncodedValue: function(value) {
|
urlEncodedValue: function(value) {
|
||||||
var
|
// GITEA-PATCH: always encode the value.
|
||||||
decodedValue = window.decodeURIComponent(value),
|
// Old code does "decodeURIComponent" first to guess whether the value is encoded, it is not right.
|
||||||
encodedValue = window.encodeURIComponent(value),
|
return window.encodeURIComponent(value);
|
||||||
alreadyEncoded = (decodedValue !== value)
|
|
||||||
;
|
|
||||||
if(alreadyEncoded) {
|
|
||||||
module.debug('URL value is already encoded, avoiding double encoding', value);
|
|
||||||
return value;
|
|
||||||
}
|
|
||||||
module.verbose('Encoding value using encodeURIComponent', value, encodedValue);
|
|
||||||
return encodedValue;
|
|
||||||
},
|
},
|
||||||
defaultData: function() {
|
defaultData: function() {
|
||||||
var
|
var
|
||||||
|
@ -31,9 +31,9 @@ $.fn.dropdown = function(parameters) {
|
|||||||
moduleSelector = $allModules.selector || '',
|
moduleSelector = $allModules.selector || '',
|
||||||
|
|
||||||
hasTouch = ('ontouchstart' in document.documentElement),
|
hasTouch = ('ontouchstart' in document.documentElement),
|
||||||
clickEvent = hasTouch
|
// GITEA-PATCH: always "click" as clickEvent, old code used "touchstart" as clickEvent, it is wrong,
|
||||||
? 'touchstart'
|
// because "touchstart" caused problems when users try to scroll and the touch point is in the dropdown.
|
||||||
: 'click',
|
clickEvent = 'click',
|
||||||
|
|
||||||
time = new Date().getTime(),
|
time = new Date().getTime(),
|
||||||
performance = [],
|
performance = [],
|
||||||
@ -768,7 +768,7 @@ $.fn.dropdown = function(parameters) {
|
|||||||
preSelected = preSelected && preSelected!=="" ? preSelected.split(settings.delimiter) : [];
|
preSelected = preSelected && preSelected!=="" ? preSelected.split(settings.delimiter) : [];
|
||||||
}
|
}
|
||||||
$.each(preSelected,function(index,value){
|
$.each(preSelected,function(index,value){
|
||||||
$item.filter('[data-value="'+value+'"]')
|
$item.filter('[data-value="'+CSS.escape(value)+'"]') // GITEA-PATCH: use "CSS.escape" for query selector
|
||||||
.addClass(className.filtered)
|
.addClass(className.filtered)
|
||||||
;
|
;
|
||||||
});
|
});
|
||||||
@ -1027,7 +1027,7 @@ $.fn.dropdown = function(parameters) {
|
|||||||
$input.append('<option disabled selected value></option>');
|
$input.append('<option disabled selected value></option>');
|
||||||
$.each(values, function(index, item) {
|
$.each(values, function(index, item) {
|
||||||
var
|
var
|
||||||
value = settings.templates.deQuote(item[fields.value]),
|
value = settings.templates.escape(item[fields.value]), // GITEA-PATCH: use "escape" for attribute value
|
||||||
name = settings.templates.escape(
|
name = settings.templates.escape(
|
||||||
item[fields.name] || '',
|
item[fields.name] || '',
|
||||||
settings.preserveHTML
|
settings.preserveHTML
|
||||||
@ -4180,13 +4180,14 @@ $.fn.dropdown.settings.templates = {
|
|||||||
if( itemType === 'item' ) {
|
if( itemType === 'item' ) {
|
||||||
var
|
var
|
||||||
maybeText = (option[fields.text])
|
maybeText = (option[fields.text])
|
||||||
? ' data-text="' + deQuote(option[fields.text]) + '"'
|
? ' data-text="' + escape(option[fields.text]) + '"' // GITEA-PATCH: use "escape" for attribute value
|
||||||
: '',
|
: '',
|
||||||
maybeDisabled = (option[fields.disabled])
|
maybeDisabled = (option[fields.disabled])
|
||||||
? className.disabled+' '
|
? className.disabled+' '
|
||||||
: ''
|
: ''
|
||||||
;
|
;
|
||||||
html += '<div class="'+ maybeDisabled + (option[fields.class] ? deQuote(option[fields.class]) : className.item)+'" data-value="' + deQuote(option[fields.value]) + '"' + maybeText + '>';
|
// GITEA-PATCH: use "escape" for attribute value
|
||||||
|
html += '<div class="'+ maybeDisabled + (option[fields.class] ? deQuote(option[fields.class]) : className.item)+'" data-value="' + escape(option[fields.value]) + '"' + maybeText + '>';
|
||||||
if(option[fields.image]) {
|
if(option[fields.image]) {
|
||||||
html += '<img class="'+(option[fields.imageClass] ? deQuote(option[fields.imageClass]) : className.image)+'" src="' + deQuote(option[fields.image]) + '">';
|
html += '<img class="'+(option[fields.imageClass] ? deQuote(option[fields.imageClass]) : className.image)+'" src="' + deQuote(option[fields.image]) + '">';
|
||||||
}
|
}
|
||||||
|
@ -3,8 +3,4 @@ import './components/dropdown.js';
|
|||||||
import './components/modal.js';
|
import './components/modal.js';
|
||||||
import './components/search.js';
|
import './components/search.js';
|
||||||
|
|
||||||
// Hard forked from Fomantic 2.8.7
|
// Hard-forked from Fomantic UI 2.8.7, patches are commented with "GITEA-PATCH"
|
||||||
|
|
||||||
// TODO: need to apply the patch from Makefile
|
|
||||||
// # fomantic uses "touchstart" as click event for some browsers, it's not ideal, so we force fomantic to always use "click" as click event
|
|
||||||
// $(SED_INPLACE) -e 's/clickEvent[ \t]*=/clickEvent = "click", unstableClickEvent =/g' $(FOMANTIC_WORK_DIR)/build/semantic.js
|
|
||||||
|
@ -196,7 +196,9 @@ function initRepoDiffShowMore() {
|
|||||||
const resp = await response.text();
|
const resp = await response.text();
|
||||||
const respDoc = parseDom(resp, 'text/html');
|
const respDoc = parseDom(resp, 'text/html');
|
||||||
const respFileBody = respDoc.querySelector('#diff-file-boxes .diff-file-body .file-body');
|
const respFileBody = respDoc.querySelector('#diff-file-boxes .diff-file-body .file-body');
|
||||||
el.parentElement.replaceWith(...Array.from(respFileBody.children));
|
const respFileBodyChildren = Array.from(respFileBody.children); // respFileBody.children will be empty after replaceWith
|
||||||
|
el.parentElement.replaceWith(...respFileBodyChildren);
|
||||||
|
for (const el of respFileBodyChildren) window.htmx.process(el);
|
||||||
// FIXME: calling onShowMoreFiles is not quite right here.
|
// FIXME: calling onShowMoreFiles is not quite right here.
|
||||||
// But since onShowMoreFiles mixes "init diff box" and "init diff body" together,
|
// But since onShowMoreFiles mixes "init diff box" and "init diff body" together,
|
||||||
// so it still needs to call it to make the "ImageDiff" and something similar work.
|
// so it still needs to call it to make the "ImageDiff" and something similar work.
|
||||||
|
1
web_src/js/globals.d.ts
vendored
1
web_src/js/globals.d.ts
vendored
@ -64,6 +64,7 @@ interface Window {
|
|||||||
jQuery: typeof import('@types/jquery'),
|
jQuery: typeof import('@types/jquery'),
|
||||||
htmx: Omit<typeof import('htmx.org/dist/htmx.esm.js').default, 'config'> & {
|
htmx: Omit<typeof import('htmx.org/dist/htmx.esm.js').default, 'config'> & {
|
||||||
config?: Writable<typeof import('htmx.org').default.config>,
|
config?: Writable<typeof import('htmx.org').default.config>,
|
||||||
|
process?: (elt: Element | string) => void,
|
||||||
},
|
},
|
||||||
_globalHandlerErrors: Array<ErrorEvent & PromiseRejectionEvent> & {
|
_globalHandlerErrors: Array<ErrorEvent & PromiseRejectionEvent> & {
|
||||||
_inited: boolean,
|
_inited: boolean,
|
||||||
|
@ -1,5 +1,4 @@
|
|||||||
import $ from 'jquery';
|
import $ from 'jquery';
|
||||||
import {initFomanticApiPatch} from './fomantic/api.ts';
|
|
||||||
import {initAriaCheckboxPatch} from './fomantic/checkbox.ts';
|
import {initAriaCheckboxPatch} from './fomantic/checkbox.ts';
|
||||||
import {initAriaFormFieldPatch} from './fomantic/form.ts';
|
import {initAriaFormFieldPatch} from './fomantic/form.ts';
|
||||||
import {initAriaDropdownPatch} from './fomantic/dropdown.ts';
|
import {initAriaDropdownPatch} from './fomantic/dropdown.ts';
|
||||||
@ -27,7 +26,6 @@ export function initGiteaFomantic() {
|
|||||||
initFomanticTransition();
|
initFomanticTransition();
|
||||||
initFomanticDimmer();
|
initFomanticDimmer();
|
||||||
initFomanticTab();
|
initFomanticTab();
|
||||||
initFomanticApiPatch();
|
|
||||||
|
|
||||||
// Use the patches to improve accessibility, these patches are designed to be as independent as possible, make it easy to modify or remove in the future.
|
// Use the patches to improve accessibility, these patches are designed to be as independent as possible, make it easy to modify or remove in the future.
|
||||||
initAriaCheckboxPatch();
|
initAriaCheckboxPatch();
|
||||||
|
@ -1,41 +0,0 @@
|
|||||||
import $ from 'jquery';
|
|
||||||
import type {FomanticInitFunction} from '../../types.ts';
|
|
||||||
|
|
||||||
export function initFomanticApiPatch() {
|
|
||||||
//
|
|
||||||
// Fomantic API module has some very buggy behaviors:
|
|
||||||
//
|
|
||||||
// If encodeParameters=true, it calls `urlEncodedValue` to encode the parameter.
|
|
||||||
// However, `urlEncodedValue` just tries to "guess" whether the parameter is already encoded, by decoding the parameter and encoding it again.
|
|
||||||
//
|
|
||||||
// There are 2 problems:
|
|
||||||
// 1. It may guess wrong, and skip encoding a parameter which looks like encoded.
|
|
||||||
// 2. If the parameter can't be decoded, `decodeURIComponent` will throw an error, and the whole request will fail.
|
|
||||||
//
|
|
||||||
// This patch only fixes the second error behavior at the moment.
|
|
||||||
//
|
|
||||||
const patchKey = '_giteaFomanticApiPatch';
|
|
||||||
const oldApi = $.api;
|
|
||||||
$.api = $.fn.api = function(...args: Parameters<FomanticInitFunction>) {
|
|
||||||
const apiCall = oldApi.bind(this);
|
|
||||||
const ret = oldApi.apply(this, args);
|
|
||||||
|
|
||||||
if (typeof args[0] !== 'string') {
|
|
||||||
const internalGet = apiCall('internal', 'get');
|
|
||||||
if (!internalGet.urlEncodedValue[patchKey]) {
|
|
||||||
const oldUrlEncodedValue = internalGet.urlEncodedValue;
|
|
||||||
internalGet.urlEncodedValue = function (value: any) {
|
|
||||||
try {
|
|
||||||
return oldUrlEncodedValue(value);
|
|
||||||
} catch {
|
|
||||||
// if Fomantic API module's `urlEncodedValue` throws an error, we encode it by ourselves.
|
|
||||||
return encodeURIComponent(value);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
internalGet.urlEncodedValue[patchKey] = true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return ret;
|
|
||||||
};
|
|
||||||
$.api.settings = oldApi.settings;
|
|
||||||
}
|
|
Reference in New Issue
Block a user