移动目录

This commit is contained in:
CrescentLeaf
2025-11-23 13:27:15 +08:00
parent f13623f4fc
commit 1cb8ac3fff
479 changed files with 49 additions and 49 deletions

View File

@@ -0,0 +1 @@
export declare const navigationBarItemStyle: import("lit").CSSResult;

View File

@@ -0,0 +1,2 @@
import { css } from 'lit';
export const navigationBarItemStyle = css `:host{--shape-corner-indicator:var(--mdui-shape-corner-full);position:relative;z-index:0;flex:1;overflow:hidden;min-width:3rem;--mdui-comp-ripple-state-layer-color:var(--mdui-color-on-surface-variant)}.container{display:flex;flex-direction:column;align-items:center;justify-content:center;height:100%;text-decoration:none;cursor:pointer;-webkit-user-select:none;user-select:none;-webkit-tap-highlight-color:transparent;padding-top:.75rem;padding-bottom:.75rem}.container:not(.initial){transition:padding var(--mdui-motion-duration-short4) var(--mdui-motion-easing-standard)}mdui-ripple{z-index:1;left:50%;transform:translateX(-50%);width:4rem;height:2rem;margin-top:.75rem;border-radius:var(--mdui-shape-corner-full)}mdui-ripple:not(.initial){transition:margin-top var(--mdui-motion-duration-short4) var(--mdui-motion-easing-standard)}.indicator{position:relative;display:flex;align-items:center;justify-content:center;background-color:transparent;border-radius:var(--shape-corner-indicator);height:2rem;width:2rem}:not(.initial) .indicator{transition:background-color var(--mdui-motion-duration-short1) var(--mdui-motion-easing-standard),width var(--mdui-motion-duration-short4) var(--mdui-motion-easing-standard)}::slotted([slot=badge]){position:absolute;transform:translate(50%,-50%)}::slotted([slot=badge][variant=small]){transform:translate(.5625rem,-.5625rem)}.active-icon,.icon{color:rgb(var(--mdui-color-on-surface-variant));font-size:1.5rem}.active-icon mdui-icon,.icon mdui-icon,::slotted([slot=active]),::slotted([slot=icon]){font-size:inherit}.icon{display:flex}.active-icon{display:none}.label{display:flex;align-items:center;height:1rem;color:rgb(var(--mdui-color-on-surface-variant));margin-top:.25rem;margin-bottom:.25rem;font-size:var(--mdui-typescale-label-medium-size);font-weight:var(--mdui-typescale-label-medium-weight);letter-spacing:var(--mdui-typescale-label-medium-tracking);line-height:var(--mdui-typescale-label-medium-line-height)}:not(.initial) .label{transition:opacity var(--mdui-motion-duration-short4) var(--mdui-motion-easing-linear)}:host(:not([active])) mdui-ripple.label-visibility-selected,mdui-ripple.label-visibility-unlabeled{margin-top:1.5rem}.container.label-visibility-unlabeled,:host(:not([active])) .container.label-visibility-selected{padding-top:1.5rem;padding-bottom:0}.container.label-visibility-unlabeled .label,:host(:not([active])) .container.label-visibility-selected .label{opacity:0}:host([active]){--mdui-comp-ripple-state-layer-color:var(--mdui-color-on-surface)}:host([active]) .indicator{width:4rem;background-color:rgb(var(--mdui-color-secondary-container))}:host([active]) .active-icon,:host([active]) .icon{color:rgb(var(--mdui-color-on-secondary-container))}:host([active]) .has-active-icon .active-icon{display:flex}:host([active]) .has-active-icon .icon{display:none}:host([active]) .label{color:rgb(var(--mdui-color-on-surface))}`;

View File

@@ -0,0 +1,80 @@
import { MduiElement } from '@mdui/shared/base/mdui-element.js';
import '../icon.js';
import type { Ripple } from '../ripple/index.js';
import type { CSSResultGroup, TemplateResult } from 'lit';
declare const NavigationBarItem_base: import("@lit/reactive-element/decorators/base.js").Constructor<import("@mdui/shared/mixins/anchor.js").AnchorMixinInterface> & import("@lit/reactive-element/decorators/base.js").Constructor<import("../ripple/ripple-mixin.js").RippleMixinInterface> & import("@lit/reactive-element/decorators/base.js").Constructor<import("@mdui/shared/mixins/focusable.js").FocusableMixinInterface> & typeof MduiElement;
/**
* @summary 底部导航栏项组件。需配合 `<mdui-navigation-bar>` 组件使用
*
* ```html
* <mdui-navigation-bar>
* ..<mdui-navigation-bar-item icon="place">Item 1</mdui-navigation-bar-item>
* ..<mdui-navigation-bar-item icon="commute">Item 2</mdui-navigation-bar-item>
* ..<mdui-navigation-bar-item icon="people">Item 3</mdui-navigation-bar-item>
* </mdui-navigation-bar>
* ```
*
* @slot - 导航项文本
* @slot icon - 图标
* @slot active-icon - 激活状态的图标元素
* @slot badge - 徽标
*
* @event focus - 获得焦点时触发
* @event blur - 失去焦点时触发
*
* @csspart container - 导航项容器
* @csspart indicator - 指示器
* @csspart badge - 徽标
* @csspart icon - 图标
* @csspart active-icon - 激活状态的图标
* @csspart label - 导航项文本
*
* @cssprop --shape-corner-indicator - 指示器的圆角大小。可以指定一个具体的像素值;但更推荐引用[设计令牌](/docs/2/styles/design-tokens#shape-corner)
*/
export declare class NavigationBarItem extends NavigationBarItem_base<NavigationBarItemEventMap> {
static styles: CSSResultGroup;
/**
* 未激活状态的 Material Icons 图标名。也可以通过 `slot="icon"` 设置
*/
icon?: string;
/**
* 激活状态的 Material Icons 图标名。也可以通过 `slot="active-icon"` 设置
*/
activeIcon?: string;
/**
* 导航项的值
*/
value?: string;
/**
* 文本的可视状态,由 `<mdui-navigation-bar>` 组件控制该参数
*/
protected labelVisibility?: 'selected' | 'labeled' | 'unlabeled';
/**
* 是否是初始状态,不显示动画。由 `<mdui-navigation-bar>` 组件控制该参数
*/
protected isInitial: boolean;
/**
* 是否为激活状态,由 `<mdui-navigation-bar>` 组件控制该参数
*/
protected active: boolean;
private disabled;
protected readonly key: number;
private readonly rippleRef;
private readonly hasSlotController;
protected get rippleElement(): Ripple;
protected get rippleDisabled(): boolean;
protected get focusElement(): HTMLElement | null;
protected get focusDisabled(): boolean;
protected render(): TemplateResult;
private renderInner;
}
export interface NavigationBarItemEventMap {
focus: FocusEvent;
blur: FocusEvent;
}
declare global {
interface HTMLElementTagNameMap {
'mdui-navigation-bar-item': NavigationBarItem;
}
}
export {};

View File

@@ -0,0 +1,137 @@
import { __decorate } from "tslib";
import { html } from 'lit';
import { customElement, property, state } from 'lit/decorators.js';
import { createRef, ref } from 'lit/directives/ref.js';
import cc from 'classcat';
import { MduiElement } from '@mdui/shared/base/mdui-element.js';
import { HasSlotController } from '@mdui/shared/controllers/has-slot.js';
import { booleanConverter } from '@mdui/shared/helpers/decorator.js';
import { nothingTemplate } from '@mdui/shared/helpers/template.js';
import { uniqueId } from '@mdui/shared/helpers/uniqueId.js';
import { componentStyle } from '@mdui/shared/lit-styles/component-style.js';
import { AnchorMixin } from '@mdui/shared/mixins/anchor.js';
import { FocusableMixin } from '@mdui/shared/mixins/focusable.js';
import '../icon.js';
import { RippleMixin } from '../ripple/ripple-mixin.js';
import { navigationBarItemStyle } from './navigation-bar-item-style.js';
/**
* @summary 底部导航栏项组件。需配合 `<mdui-navigation-bar>` 组件使用
*
* ```html
* <mdui-navigation-bar>
* ..<mdui-navigation-bar-item icon="place">Item 1</mdui-navigation-bar-item>
* ..<mdui-navigation-bar-item icon="commute">Item 2</mdui-navigation-bar-item>
* ..<mdui-navigation-bar-item icon="people">Item 3</mdui-navigation-bar-item>
* </mdui-navigation-bar>
* ```
*
* @slot - 导航项文本
* @slot icon - 图标
* @slot active-icon - 激活状态的图标元素
* @slot badge - 徽标
*
* @event focus - 获得焦点时触发
* @event blur - 失去焦点时触发
*
* @csspart container - 导航项容器
* @csspart indicator - 指示器
* @csspart badge - 徽标
* @csspart icon - 图标
* @csspart active-icon - 激活状态的图标
* @csspart label - 导航项文本
*
* @cssprop --shape-corner-indicator - 指示器的圆角大小。可以指定一个具体的像素值;但更推荐引用[设计令牌](/docs/2/styles/design-tokens#shape-corner)
*/
let NavigationBarItem = class NavigationBarItem extends AnchorMixin(RippleMixin(FocusableMixin(MduiElement))) {
constructor() {
super(...arguments);
/**
* 是否是初始状态,不显示动画。由 `<mdui-navigation-bar>` 组件控制该参数
*/
this.isInitial = true;
/**
* 是否为激活状态,由 `<mdui-navigation-bar>` 组件控制该参数
*/
this.active = false;
// 是否禁用该元素,该组件没有禁用状态
this.disabled = false;
// 每一个 `<navigation-bar-item>` 元素都添加一个唯一的 key
this.key = uniqueId();
this.rippleRef = createRef();
this.hasSlotController = new HasSlotController(this, 'active-icon');
}
get rippleElement() {
return this.rippleRef.value;
}
get rippleDisabled() {
return this.disabled;
}
get focusElement() {
return this.href ? this.renderRoot?.querySelector('._a') : this;
}
get focusDisabled() {
return this.disabled;
}
render() {
const labelVisibilityClassName = cc({
'label-visibility-selected': this.labelVisibility === 'selected',
'label-visibility-labeled': this.labelVisibility === 'labeled',
'label-visibility-unlabeled': this.labelVisibility === 'unlabeled',
initial: this.isInitial,
});
const className = cc([
{
container: true,
'has-active-icon': this.activeIcon || this.hasSlotController.test('active-icon'),
},
labelVisibilityClassName,
]);
return html `<mdui-ripple .noRipple="${!this.active || this.noRipple}" class="${labelVisibilityClassName}" ${ref(this.rippleRef)}></mdui-ripple>${this.href
? this.renderAnchor({
part: 'container',
className,
content: this.renderInner(),
})
: html `<div part="container" class="${className}">${this.renderInner()}</div>`}`;
}
renderInner() {
return html `<div part="indicator" class="indicator"><slot name="badge" part="badge" class="badge"></slot><slot name="active-icon" part="active-icon" class="active-icon">${this.activeIcon
? html `<mdui-icon name="${this.activeIcon}"></mdui-icon>`
: nothingTemplate}</slot><slot name="icon" part="icon" class="icon">${this.icon
? html `<mdui-icon name="${this.icon}"></mdui-icon>`
: nothingTemplate}</slot></div><slot part="label" class="label"></slot>`;
}
};
NavigationBarItem.styles = [
componentStyle,
navigationBarItemStyle,
];
__decorate([
property({ reflect: true })
], NavigationBarItem.prototype, "icon", void 0);
__decorate([
property({ reflect: true, attribute: 'active-icon' })
], NavigationBarItem.prototype, "activeIcon", void 0);
__decorate([
property({ reflect: true })
], NavigationBarItem.prototype, "value", void 0);
__decorate([
state()
], NavigationBarItem.prototype, "labelVisibility", void 0);
__decorate([
state()
], NavigationBarItem.prototype, "isInitial", void 0);
__decorate([
property({
type: Boolean,
reflect: true,
converter: booleanConverter,
})
], NavigationBarItem.prototype, "active", void 0);
__decorate([
state()
], NavigationBarItem.prototype, "disabled", void 0);
NavigationBarItem = __decorate([
customElement('mdui-navigation-bar-item')
], NavigationBarItem);
export { NavigationBarItem };

View File

@@ -0,0 +1 @@
export declare const navigationBarStyle: import("lit").CSSResult;

View File

@@ -0,0 +1,2 @@
import { css } from 'lit';
export const navigationBarStyle = css `:host{--shape-corner:var(--mdui-shape-corner-none);--z-index:2000;position:fixed;right:0;bottom:0;left:0;display:flex;flex:0 0 auto;overflow:hidden;border-radius:var(--shape-corner) var(--shape-corner) 0 0;z-index:var(--z-index);transition-property:transform;transition-duration:var(--mdui-motion-duration-long2);transition-timing-function:var(--mdui-motion-easing-emphasized);height:5rem;background-color:rgb(var(--mdui-color-surface));box-shadow:var(--mdui-elevation-level2)}:host([scroll-target]:not([scroll-target=''])){position:absolute}:host([hide]:not([hide=false i])){transform:translateY(5.625rem);transition-duration:var(--mdui-motion-duration-short4)}`;

View File

@@ -0,0 +1,86 @@
import { PropertyValues } from 'lit';
import { LayoutItemBase } from '../layout/layout-item-base.js';
import type { LayoutPlacement } from '../layout/helper.js';
import type { ScrollPaddingPosition } from '@mdui/shared/mixins/scrollBehavior.js';
import type { CSSResultGroup, TemplateResult } from 'lit';
declare const NavigationBar_base: import("@lit/reactive-element/decorators/base.js").Constructor<import("@mdui/shared/mixins/scrollBehavior.js").ScrollBehaviorMixinInterface> & typeof LayoutItemBase;
/**
* @summary 底部导航栏组件。需配合 `<mdui-navigation-bar-item>` 组件使用
*
* ```html
* <mdui-navigation-bar>
* ..<mdui-navigation-bar-item icon="place">Item 1</mdui-navigation-bar-item>
* ..<mdui-navigation-bar-item icon="commute">Item 2</mdui-navigation-bar-item>
* ..<mdui-navigation-bar-item icon="people">Item 3</mdui-navigation-bar-item>
* </mdui-navigation-bar>
* ```
*
* @event change - 值变化时触发
* @event show - 开始显示时,事件被触发。可以通过调用 `event.preventDefault()` 阻止显示
* @event shown - 显示动画完成时,事件被触发
* @event hide - 开始隐藏时,事件被触发。可以通过调用 `event.preventDefault()` 阻止隐藏
* @event hidden - 隐藏动画完成时,事件被触发
*
* @slot - `<mdui-navigation-bar-item>` 组件
*
* @cssprop --shape-corner - 组件的圆角大小。可以指定一个具体的像素值;但更推荐引用[设计令牌](/docs/2/styles/design-tokens#shape-corner)
* @cssprop --z-index - 组件的 CSS `z-index` 值
*/
export declare class NavigationBar extends NavigationBar_base<NavigationBarEventMap> {
static styles: CSSResultGroup;
/**
* 是否隐藏
*/
hide: boolean;
/**
* 文本的可视状态。可选值包括:
*
* * `auto`当选项小于等于3个时始终显示文本当选项大于3个时仅显示选中状态的文本
* * `selected`:仅在选中状态显示文本
* * `labeled`:始终显示文本
* * `unlabeled`:始终不显示文本
*/
labelVisibility: /*当选项小于等于3个时始终显示文本当选项大于3个时仅显示选中状态的文本*/ 'auto' | /*仅在选中状态显示文本*/ 'selected' | /*始终显示文本*/ 'labeled' | /*始终不显示文本*/ 'unlabeled';
/**
* 当前选中的 `<mdui-navigation-bar-item>` 的值
*/
value?: string;
/**
* 滚动行为。可选值包括:
*
* * `hide`:滚动时隐藏
*/
scrollBehavior?: 'hide';
private activeKey;
private readonly items;
private isInitial;
private definedController;
protected get scrollPaddingPosition(): ScrollPaddingPosition;
protected get layoutPlacement(): LayoutPlacement;
private onActiveKeyChange;
private onValueChange;
private onLabelVisibilityChange;
protected firstUpdated(_changedProperties: PropertyValues): void;
protected render(): TemplateResult;
/**
* 滚动行为
* 当前仅支持 hide 这一个行为,所以不做行为类型判断
*/
protected runScrollThreshold(isScrollingUp: boolean): void;
private onClick;
private updateItems;
private onSlotChange;
}
export interface NavigationBarEventMap {
change: CustomEvent<void>;
show: CustomEvent<void>;
shown: CustomEvent<void>;
hide: CustomEvent<void>;
hidden: CustomEvent<void>;
}
declare global {
interface HTMLElementTagNameMap {
'mdui-navigation-bar': NavigationBar;
}
}
export {};

View File

@@ -0,0 +1,189 @@
import { __decorate } from "tslib";
import { html } from 'lit';
import { customElement, property, queryAssignedElements, state, } from 'lit/decorators.js';
import { DefinedController } from '@mdui/shared/controllers/defined.js';
import { watch } from '@mdui/shared/decorators/watch.js';
import { booleanConverter } from '@mdui/shared/helpers/decorator.js';
import { componentStyle } from '@mdui/shared/lit-styles/component-style.js';
import { ScrollBehaviorMixin } from '@mdui/shared/mixins/scrollBehavior.js';
import { LayoutItemBase } from '../layout/layout-item-base.js';
import { navigationBarStyle } from './navigation-bar-style.js';
/**
* @summary 底部导航栏组件。需配合 `<mdui-navigation-bar-item>` 组件使用
*
* ```html
* <mdui-navigation-bar>
* ..<mdui-navigation-bar-item icon="place">Item 1</mdui-navigation-bar-item>
* ..<mdui-navigation-bar-item icon="commute">Item 2</mdui-navigation-bar-item>
* ..<mdui-navigation-bar-item icon="people">Item 3</mdui-navigation-bar-item>
* </mdui-navigation-bar>
* ```
*
* @event change - 值变化时触发
* @event show - 开始显示时,事件被触发。可以通过调用 `event.preventDefault()` 阻止显示
* @event shown - 显示动画完成时,事件被触发
* @event hide - 开始隐藏时,事件被触发。可以通过调用 `event.preventDefault()` 阻止隐藏
* @event hidden - 隐藏动画完成时,事件被触发
*
* @slot - `<mdui-navigation-bar-item>` 组件
*
* @cssprop --shape-corner - 组件的圆角大小。可以指定一个具体的像素值;但更推荐引用[设计令牌](/docs/2/styles/design-tokens#shape-corner)
* @cssprop --z-index - 组件的 CSS `z-index` 值
*/
let NavigationBar = class NavigationBar extends ScrollBehaviorMixin(LayoutItemBase) {
constructor() {
super(...arguments);
/**
* 是否隐藏
*/
this.hide = false;
/**
* 文本的可视状态。可选值包括:
*
* * `auto`当选项小于等于3个时始终显示文本当选项大于3个时仅显示选中状态的文本
* * `selected`:仅在选中状态显示文本
* * `labeled`:始终显示文本
* * `unlabeled`:始终不显示文本
*/
this.labelVisibility = 'auto';
// 因为 navigation-bar-item 的 value 可能会重复,所以在每个 navigation-bar-item 元素上都添加了一个唯一的 key通过 activeKey 来记录激活状态的 key
this.activeKey = 0;
// 是否是初始状态,初始状态不触发 change 事件,没有动画
this.isInitial = true;
this.definedController = new DefinedController(this, {
relatedElements: ['mdui-navigation-bar-item'],
});
}
get scrollPaddingPosition() {
return 'bottom';
}
get layoutPlacement() {
return 'bottom';
}
async onActiveKeyChange() {
await this.definedController.whenDefined();
// 根据 activeKey 读取对应 navigation-bar-item 的值
const item = this.items.find((item) => item.key === this.activeKey);
this.value = item?.value;
if (!this.isInitial) {
this.emit('change');
}
}
async onValueChange() {
this.isInitial = !this.hasUpdated;
await this.definedController.whenDefined();
const item = this.items.find((item) => item.value === this.value);
this.activeKey = item?.key ?? 0;
this.updateItems();
}
async onLabelVisibilityChange() {
await this.definedController.whenDefined();
this.updateItems();
}
firstUpdated(_changedProperties) {
super.firstUpdated(_changedProperties);
this.addEventListener('transitionend', (event) => {
if (event.target === this) {
this.emit(this.hide ? 'hidden' : 'shown');
}
});
}
render() {
return html `<slot @slotchange="${this.onSlotChange}" @click="${this.onClick}"></slot>`;
}
/**
* 滚动行为
* 当前仅支持 hide 这一个行为,所以不做行为类型判断
*/
runScrollThreshold(isScrollingUp) {
// 向下滚动
if (!isScrollingUp && !this.hide) {
const eventProceeded = this.emit('hide', { cancelable: true });
if (eventProceeded) {
this.hide = true;
}
}
// 向上滚动
if (isScrollingUp && this.hide) {
const eventProceeded = this.emit('show', { cancelable: true });
if (eventProceeded) {
this.hide = false;
}
}
}
onClick(event) {
// event.button 为 0 时,为鼠标左键点击。忽略鼠标中键和右键
if (event.button) {
return;
}
const target = event.target;
const item = target.closest('mdui-navigation-bar-item');
if (!item) {
return;
}
this.activeKey = item.key;
this.isInitial = false;
this.updateItems();
}
// 更新 <mdui-navigation-bar-item> 的状态
updateItems() {
const items = this.items;
// <mdui-navigation-bar-item> 的 labelVisibility 不含 auto
const labelVisibility = this.labelVisibility === 'auto'
? items.length <= 3
? 'labeled'
: 'selected'
: this.labelVisibility;
items.forEach((item) => {
item.active = this.activeKey === item.key;
item.labelVisibility = labelVisibility;
item.isInitial = this.isInitial;
});
}
async onSlotChange() {
await this.definedController.whenDefined();
this.updateItems();
}
};
NavigationBar.styles = [
componentStyle,
navigationBarStyle,
];
__decorate([
property({
type: Boolean,
reflect: true,
converter: booleanConverter,
})
], NavigationBar.prototype, "hide", void 0);
__decorate([
property({ reflect: true, attribute: 'label-visibility' })
], NavigationBar.prototype, "labelVisibility", void 0);
__decorate([
property({ reflect: true })
], NavigationBar.prototype, "value", void 0);
__decorate([
property({ reflect: true, attribute: 'scroll-behavior' })
], NavigationBar.prototype, "scrollBehavior", void 0);
__decorate([
state()
], NavigationBar.prototype, "activeKey", void 0);
__decorate([
queryAssignedElements({
selector: 'mdui-navigation-bar-item',
flatten: true,
})
], NavigationBar.prototype, "items", void 0);
__decorate([
watch('activeKey', true)
], NavigationBar.prototype, "onActiveKeyChange", null);
__decorate([
watch('value')
], NavigationBar.prototype, "onValueChange", null);
__decorate([
watch('labelVisibility', true)
], NavigationBar.prototype, "onLabelVisibilityChange", null);
NavigationBar = __decorate([
customElement('mdui-navigation-bar')
], NavigationBar);
export { NavigationBar };