Skip to content

Commit

Permalink
feat(core): optimize url validator & popup on selection moving (#2845)
Browse files Browse the repository at this point in the history
Co-authored-by: tonytonychopper123 <40351140+tonytonychopper123@users.noreply.github.com>
  • Loading branch information
weird94 and tonytonychopper123 authored Jul 24, 2024
1 parent ae0b012 commit c6ab2a7
Show file tree
Hide file tree
Showing 6 changed files with 328 additions and 10 deletions.
3 changes: 2 additions & 1 deletion packages/core/src/common/__tests__/url.spec.ts
Original file line number Diff line number Diff line change
Expand Up @@ -27,7 +27,8 @@ describe('Test url utils', () => {
expect(isLegalUrl('https://univer')).toBeFalsy();
expect(isLegalUrl('univer')).toBeFalsy();
expect(isLegalUrl('192.168')).toBeFalsy();
expect(isLegalUrl('咋看手机看到. 是多少')).toBeFalsy();
expect(isLegalUrl('咋看手机看到.是多少')).toBeFalsy();
expect(isLegalUrl('univer.univer')).toBeFalsy();
});

it('should add protocol to no protocol url', () => {
Expand Down
293 changes: 292 additions & 1 deletion packages/core/src/common/url.ts
Original file line number Diff line number Diff line change
Expand Up @@ -14,6 +14,278 @@
* limitations under the License.
*/

const topLevelDomainSet = new Set([
'ac',
'ad',
'ae',
'aero',
'af',
'ag',
'ai',
'al',
'am',
'ao',
'aq',
'ar',
'arpa',
'as',
'asia',
'at',
'au',
'aw',
'ax',
'az',
'ba',
'bb',
'bd',
'be',
'bf',
'bg',
'bh',
'bi',
'biz',
'bj',
'bm',
'bn',
'bo',
'br',
'bs',
'bt',
'bv',
'bw',
'by',
'bz',
'ca',
'cat',
'cc',
'cd',
'cf',
'cg',
'ch',
'ci',
'ck',
'cl',
'cm',
'cn',
'co',
'com',
'coop',
'cr',
'cu',
'cv',
'cw',
'cx',
'cy',
'cz',
'de',
'dj',
'dk',
'dm',
'do',
'dz',
'ec',
'edu',
'ee',
'eg',
'er',
'es',
'et',
'eu',
'fi',
'fj',
'fk',
'fm',
'fo',
'fr',
'ga',
'gb',
'gd',
'ge',
'gf',
'gg',
'gh',
'gi',
'gl',
'gm',
'gn',
'gov',
'gp',
'gq',
'gr',
'gs',
'gt',
'gu',
'gw',
'gy',
'hk',
'hm',
'hn',
'hr',
'ht',
'hu',
'id',
'ie',
'il',
'im',
'in',
'info',
'int',
'io',
'iq',
'ir',
'is',
'it',
'je',
'jm',
'jo',
'jobs',
'jp',
'ke',
'kg',
'kh',
'ki',
'km',
'kn',
'kp',
'kr',
'kw',
'ky',
'kz',
'la',
'lb',
'lc',
'li',
'lk',
'lr',
'ls',
'lt',
'lu',
'lv',
'ly',
'ma',
'mc',
'md',
'me',
'mg',
'mh',
'mil',
'mk',
'ml',
'mm',
'mn',
'mo',
'mobi',
'mp',
'mq',
'mr',
'ms',
'mt',
'mu',
'museum',
'mv',
'mw',
'mx',
'my',
'mz',
'na',
'name',
'nc',
'ne',
'net',
'nf',
'ng',
'ni',
'nl',
'no',
'np',
'nr',
'nu',
'nz',
'om',
'onion',
'org',
'pa',
'pe',
'pf',
'pg',
'ph',
'pk',
'pl',
'pm',
'pn',
'post',
'pr',
'pro',
'ps',
'pt',
'pw',
'py',
'qa',
're',
'ro',
'rs',
'ru',
'rw',
'sa',
'sb',
'sc',
'sd',
'se',
'sg',
'sh',
'si',
'sj',
'sk',
'sl',
'sm',
'sn',
'so',
'sr',
'ss',
'st',
'su',
'sv',
'sx',
'sy',
'sz',
'tc',
'td',
'tel',
'tf',
'tg',
'th',
'tj',
'tk',
'tl',
'tm',
'tn',
'to',
'tr',
'tt',
'tv',
'tw',
'tz',
'ua',
'ug',
'uk',
'us',
'uy',
'uz',
'va',
'vc',
've',
'vg',
'vi',
'vn',
'vu',
'wf',
'ws',
'yt',
'za',
'zm',
'zw',
]);

const re_weburl = new RegExp(
'^' +
// protocol identifier (optional)
Expand Down Expand Up @@ -64,7 +336,26 @@ export function isLegalUrl(url: string) {
if (url.startsWith('http://localhost:3002') || url.startsWith('localhost:3002')) {
return true;
}
return re_weburl.test(url);

const legal = re_weburl.test(url);

if (legal) {
if (hasProtocol(url)) {
return true;
} else {
try {
const urlObj = new URL(normalizeUrl(url));
const topLevelDomain = urlObj.hostname.split('.').pop();
if (topLevelDomain && topLevelDomainSet.has(topLevelDomain)) {
return true;
};
} catch (error) {
return false;
}
}
}

return false;
}

function hasProtocol(urlString: string) {
Expand Down
2 changes: 1 addition & 1 deletion packages/debugger/src/components/Image.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -19,5 +19,5 @@ import React from 'react';
const image = '';

export const ImageDemo = () => {
return <img src={image} style={{ width: '100%', height: '100%' }} />;
return <img draggable="false" src={image} style={{ width: '100%', height: '100%' }} />;
};
Original file line number Diff line number Diff line change
Expand Up @@ -37,7 +37,7 @@ export class SheetsHyperLinkPopupController extends Disposable {

private _initHoverListener() {
this.disposeWithMe(
this._hoverManagerService.currentCell$.pipe(debounceTime(100)).subscribe((currentCell) => {
this._hoverManagerService.currentCell$.pipe(debounceTime(200)).subscribe((currentCell) => {
if (!currentCell) {
this._sheetsHyperLinkPopupService.hideCurrentPopup();
return;
Expand Down
15 changes: 11 additions & 4 deletions packages/sheets-ui/src/services/canvas-pop-manager.service.ts
Original file line number Diff line number Diff line change
Expand Up @@ -27,6 +27,7 @@ import { getViewportByCell, transformBound2OffsetBound } from '../common/utils';
import { SetScrollOperation } from '../commands/operations/scroll.operation';
import { SetZoomRatioOperation } from '../commands/operations/set-zoom-ratio.operation';
import { SheetSkeletonManagerService } from './sheet-skeleton-manager.service';
import { ISheetSelectionRenderService } from './selection/base-selection-render.service';

export interface ICanvasPopup extends Pick<IPopup,
'direction' | 'excludeOutside' | 'closeOnSelfTarget' | 'componentKey' | 'offset' | 'onClickOutside'
Expand Down Expand Up @@ -156,24 +157,30 @@ export class SheetCanvasPopManagerService extends Disposable {
* @param viewport target viewport
* @returns disposable
*/
attachPopupToCell(row: number, col: number, popup: ICanvasPopup, viewport?: Viewport): Nullable<IDisposable> {
attachPopupToCell(row: number, col: number, popup: ICanvasPopup, viewport?: Viewport, showOnSelectionMoving = false): Nullable<IDisposable> {
const workbook = this._univerInstanceService.getCurrentUnitForType<Workbook>(UniverInstanceType.UNIVER_SHEET)!;

const worksheet = workbook.getActiveSheet();
if (!worksheet) {
return null;
}

const unitId = workbook.getUnitId();
const subUnitId = worksheet.getSheetId();
const skeleton = this._renderManagerService.getRenderById(unitId)?.with(SheetSkeletonManagerService).getOrCreateSkeleton({
const currentRender = this._renderManagerService.getRenderById(unitId);
const skeleton = currentRender?.with(SheetSkeletonManagerService).getOrCreateSkeleton({
sheetId: subUnitId,
});
const sheetSelectionRenderService = currentRender?.with(ISheetSelectionRenderService);

const currentRender = this._renderManagerService.getRenderById(unitId);
if (!currentRender || !skeleton) {
if (!currentRender || !skeleton || !sheetSelectionRenderService) {
return null;
}

if (sheetSelectionRenderService.selectionMoving && !showOnSelectionMoving) {
return;
}

const activeViewport = viewport ?? getViewportByCell(row, col, currentRender.scene, worksheet);
if (!activeViewport) {
return null;
Expand Down
Loading

0 comments on commit c6ab2a7

Please sign in to comment.