mirror of
https://github.com/ionic-team/ionic-framework.git
synced 2025-08-18 19:21:34 +08:00
328 lines
8.5 KiB
TypeScript
328 lines
8.5 KiB
TypeScript
import { OpaqueToken } from '@angular/core';
|
|
|
|
import { DeepLinkConfig, NavLink, NavSegment } from './nav-util';
|
|
import { isArray, isBlank, isPresent } from '../util/util';
|
|
|
|
|
|
/**
|
|
* @hidden
|
|
*/
|
|
export class UrlSerializer {
|
|
links: NavLink[];
|
|
|
|
constructor(config: DeepLinkConfig) {
|
|
if (config && isArray(config.links)) {
|
|
this.links = normalizeLinks(config.links);
|
|
|
|
} else {
|
|
this.links = [];
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Parse the URL into a Path, which is made up of multiple NavSegments.
|
|
* Match which components belong to each segment.
|
|
*/
|
|
parse(browserUrl: string): NavSegment[] {
|
|
if (browserUrl.charAt(0) === '/') {
|
|
browserUrl = browserUrl.substr(1);
|
|
}
|
|
|
|
// trim off data after ? and #
|
|
browserUrl = browserUrl.split('?')[0].split('#')[0];
|
|
|
|
return parseUrlParts(browserUrl.split('/'), this.links);
|
|
}
|
|
|
|
createSegmentFromName(nameOrComponent: any): NavSegment {
|
|
const configLink = this.getLinkFromName(nameOrComponent);
|
|
|
|
return configLink ? {
|
|
id: configLink.name,
|
|
name: configLink.name,
|
|
component: configLink.component,
|
|
loadChildren: configLink.loadChildren,
|
|
data: null,
|
|
defaultHistory: configLink.defaultHistory
|
|
} : null;
|
|
}
|
|
|
|
getLinkFromName(nameOrComponent: any) {
|
|
return this.links.find(link => {
|
|
return (link.component === nameOrComponent) ||
|
|
(link.name === nameOrComponent);
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Serialize a path, which is made up of multiple NavSegments,
|
|
* into a URL string. Turn each segment into a string and concat them to a URL.
|
|
*/
|
|
serialize(path: NavSegment[]): string {
|
|
return '/' + path.map(segment => segment.id).join('/');
|
|
}
|
|
|
|
/**
|
|
* Serializes a component and its data into a NavSegment.
|
|
*/
|
|
serializeComponent(component: any, data: any): NavSegment {
|
|
if (component) {
|
|
const link = findLinkByComponentData(this.links, component, data);
|
|
if (link) {
|
|
return this._createSegment(link, data);
|
|
}
|
|
}
|
|
return null;
|
|
}
|
|
|
|
/** @internal */
|
|
_createSegment(configLink: NavLink, data: any): NavSegment {
|
|
let urlParts = configLink.parts;
|
|
|
|
if (isPresent(data)) {
|
|
// create a copy of the original parts in the link config
|
|
urlParts = urlParts.slice();
|
|
|
|
// loop through all the data and convert it to a string
|
|
const keys = Object.keys(data);
|
|
const keysLength = keys.length;
|
|
|
|
if (keysLength) {
|
|
for (var i = 0; i < urlParts.length; i++) {
|
|
if (urlParts[i].charAt(0) === ':') {
|
|
for (var j = 0; j < keysLength; j++) {
|
|
if (urlParts[i] === `:${keys[j]}`) {
|
|
// this data goes into the URL part (between slashes)
|
|
urlParts[i] = encodeURIComponent(data[keys[j]]);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return {
|
|
id: urlParts.join('/'),
|
|
name: configLink.name,
|
|
component: configLink.component,
|
|
loadChildren: configLink.loadChildren,
|
|
data: data,
|
|
defaultHistory: configLink.defaultHistory
|
|
};
|
|
}
|
|
|
|
formatUrlPart(name: string): string {
|
|
name = name.replace(URL_REPLACE_REG, '-');
|
|
name = name.charAt(0).toLowerCase() + name.substring(1).replace(/[A-Z]/g, match => {
|
|
return '-' + match.toLowerCase();
|
|
});
|
|
while (name.indexOf('--') > -1) {
|
|
name = name.replace('--', '-');
|
|
}
|
|
if (name.charAt(0) === '-') {
|
|
name = name.substring(1);
|
|
}
|
|
if (name.substring(name.length - 1) === '-') {
|
|
name = name.substring(0, name.length - 1);
|
|
}
|
|
return encodeURIComponent(name);
|
|
}
|
|
|
|
}
|
|
|
|
export const parseUrlParts = (urlParts: string[], configLinks: NavLink[]): NavSegment[] => {
|
|
const configLinkLen = configLinks.length;
|
|
const urlPartsLen = urlParts.length;
|
|
const segments: NavSegment[] = new Array(urlPartsLen);
|
|
|
|
for (var i = 0; i < configLinkLen; i++) {
|
|
// compare url parts to config link parts to create nav segments
|
|
var configLink = configLinks[i];
|
|
if (configLink.partsLen <= urlPartsLen) {
|
|
fillMatchedUrlParts(segments, urlParts, configLink);
|
|
}
|
|
}
|
|
|
|
// remove all the undefined segments
|
|
for (var i = urlPartsLen - 1; i >= 0; i--) {
|
|
if (segments[i] === undefined) {
|
|
if (urlParts[i] === undefined) {
|
|
// not a used part, so remove it
|
|
segments.splice(i, 1);
|
|
|
|
} else {
|
|
// create an empty part
|
|
segments[i] = {
|
|
id: urlParts[i],
|
|
name: urlParts[i],
|
|
component: null,
|
|
loadChildren: null,
|
|
data: null
|
|
};
|
|
}
|
|
}
|
|
}
|
|
|
|
return segments;
|
|
};
|
|
|
|
export const fillMatchedUrlParts = (segments: NavSegment[], urlParts: string[], configLink: NavLink) => {
|
|
for (var i = 0; i < urlParts.length; i++) {
|
|
var urlI = i;
|
|
|
|
for (var j = 0; j < configLink.partsLen; j++) {
|
|
if (isPartMatch(urlParts[urlI], configLink.parts[j])) {
|
|
urlI++;
|
|
} else {
|
|
break;
|
|
}
|
|
}
|
|
|
|
if ((urlI - i) === configLink.partsLen) {
|
|
var matchedUrlParts = urlParts.slice(i, urlI);
|
|
for (var j = i; j < urlI; j++) {
|
|
urlParts[j] = undefined;
|
|
}
|
|
segments[i] = {
|
|
id: matchedUrlParts.join('/'),
|
|
name: configLink.name,
|
|
component: configLink.component,
|
|
loadChildren: configLink.loadChildren,
|
|
data: createMatchedData(matchedUrlParts, configLink),
|
|
defaultHistory: configLink.defaultHistory
|
|
};
|
|
}
|
|
}
|
|
};
|
|
|
|
export const isPartMatch = (urlPart: string, configLinkPart: string) => {
|
|
if (isPresent(urlPart) && isPresent(configLinkPart)) {
|
|
if (configLinkPart.charAt(0) === ':') {
|
|
return true;
|
|
}
|
|
return (urlPart === configLinkPart);
|
|
}
|
|
return false;
|
|
};
|
|
|
|
export const createMatchedData = (matchedUrlParts: string[], link: NavLink): any => {
|
|
let data: any = null;
|
|
|
|
for (var i = 0; i < link.partsLen; i++) {
|
|
if (link.parts[i].charAt(0) === ':') {
|
|
data = data || {};
|
|
data[link.parts[i].substring(1)] = decodeURIComponent(matchedUrlParts[i]);
|
|
}
|
|
}
|
|
|
|
return data;
|
|
};
|
|
|
|
export const findLinkByComponentData = (links: NavLink[], component: any, instanceData: any): NavLink => {
|
|
let foundLink: NavLink = null;
|
|
let foundLinkDataMatches = -1;
|
|
|
|
for (var i = 0; i < links.length; i++) {
|
|
var link = links[i];
|
|
if (link.component === component) {
|
|
// ok, so the component matched, but multiple links can point
|
|
// to the same component, so let's make sure this is the right link
|
|
var dataMatches = 0;
|
|
if (instanceData) {
|
|
var instanceDataKeys = Object.keys(instanceData);
|
|
|
|
// this link has data
|
|
for (var j = 0; j < instanceDataKeys.length; j++) {
|
|
if (isPresent(link.dataKeys[instanceDataKeys[j]])) {
|
|
dataMatches++;
|
|
}
|
|
}
|
|
|
|
} else if (link.dataLen) {
|
|
// this component does not have data but the link does
|
|
continue;
|
|
}
|
|
|
|
if (dataMatches >= foundLinkDataMatches) {
|
|
foundLink = link;
|
|
foundLinkDataMatches = dataMatches;
|
|
}
|
|
}
|
|
}
|
|
|
|
return foundLink;
|
|
};
|
|
|
|
export const normalizeLinks = (links: NavLink[]): NavLink[] => {
|
|
for (var i = 0, ilen = links.length; i < ilen; i++) {
|
|
var link = links[i];
|
|
|
|
if (isBlank(link.segment)) {
|
|
link.segment = link.name;
|
|
}
|
|
|
|
link.dataKeys = {};
|
|
link.parts = link.segment.split('/');
|
|
link.partsLen = link.parts.length;
|
|
|
|
// used for sorting
|
|
link.staticLen = link.dataLen = 0;
|
|
var stillCountingStatic = true;
|
|
|
|
for (var j = 0; j < link.partsLen; j++) {
|
|
if (link.parts[j].charAt(0) === ':') {
|
|
link.dataLen++;
|
|
stillCountingStatic = false;
|
|
link.dataKeys[link.parts[j].substring(1)] = true;
|
|
|
|
} else if (stillCountingStatic) {
|
|
link.staticLen++;
|
|
}
|
|
}
|
|
}
|
|
|
|
// sort by the number of parts, with the links
|
|
// with the most parts first
|
|
return links.sort(sortConfigLinks);
|
|
};
|
|
|
|
function sortConfigLinks(a: NavLink, b: NavLink) {
|
|
// sort by the number of parts
|
|
if (a.partsLen > b.partsLen) {
|
|
return -1;
|
|
}
|
|
if (a.partsLen < b.partsLen) {
|
|
return 1;
|
|
}
|
|
|
|
// sort by the number of static parts in a row
|
|
if (a.staticLen > b.staticLen) {
|
|
return -1;
|
|
}
|
|
if (a.staticLen < b.staticLen) {
|
|
return 1;
|
|
}
|
|
|
|
// sort by the number of total data parts
|
|
if (a.dataLen < b.dataLen) {
|
|
return -1;
|
|
}
|
|
if (a.dataLen > b.dataLen) {
|
|
return 1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
const URL_REPLACE_REG = /\s+|\?|\!|\$|\,|\.|\+|\"|\'|\*|\^|\||\/|\\|\[|\]|#|%|`|>|<|;|:|@|&|=/g;
|
|
|
|
/**
|
|
* @hidden
|
|
*/
|
|
export const DeepLinkConfigToken = new OpaqueToken('USERLINKS');
|
|
|
|
export function setupUrlSerializer(userDeepLinkConfig: any): UrlSerializer {
|
|
return new UrlSerializer(userDeepLinkConfig);
|
|
}
|